c语言状态机_什么是状态机?用C语言实现进程5状态模型

前言

状态机在实际工作开发中应用非常广泛,在刚进入公司的时候,根据公司产品做流程图的时候,发现自己经常会漏了这样或那样的状态,导致整体流程会有问题,后来知道了状态机这样的东西,发现用这幅图就可以很清晰的表达整个状态的流转。

一口君曾经做过很多网络协议模块,很多协议的开发都必须用到状态机;一个健壮的状态机可以让你的程序,不论发生何种突发事件都不会突然进入一个不可预知的程序分支。

本篇通过C语言实现一个简单的进程5状态模型的状态机,让大家熟悉一下状态机的魅力。

什么是状态机?

定义

状态机是有限状态自动机的简称,是现实事物运行规则抽象而成的一个数学模型。

先来解释什么是“状态”( State )。现实事物是有不同状态的,例如一个LED等,就有 亮 和 灭两种状态。我们通常所说的状态机是有限状态机,也就是被描述的事物的状态的数量是有限个,例如LED灯的状态就是两个 亮和 灭。

状态机,也就是 State Machine ,不是指一台实际机器,而是指一个数学模型。说白了,一般就是指一张状态转换图。

举例

以物理课学的灯泡图为例,就是一个最基本的小型状态机

f4f1639e94af51def3a8a509f23ccd72.png


可以画出以下的状态机图

f4f1639e94af51def3a8a509f23ccd72.png

这里就是两个状态:①灯泡亮,②灯泡灭 如果打开开关,那么状态就会切换为 灯泡亮 。灯泡亮 状态下如果关闭开关,状态就会切换为 灯泡灭。

状态机的全称是有限状态自动机,自动两个字也是包含重要含义的。给定一个状态机,同时给定它的当前状态以及输入,那么输出状态时可以明确的运算出来的。例如对于灯泡,给定初始状态灯泡灭 ,给定输入“打开开关”,那么下一个状态时可以运算出来的。

四大概念

下面来给出状态机的四大概念。

  1. State ,状态。一个状态机至少要包含两个状态。例如上面灯泡的例子,有 灯泡亮和 灯泡灭两个状态。
  2. Event ,事件。事件就是执行某个操作的触发条件或者口令。对于灯泡,“打开开关”就是一个事件。
  3. Action ,动作。事件发生以后要执行动作。例如事件是“打开开关”,动作是“开灯”。编程的时候,一个 Action 一般就对应一个函数。
  4. Transition ,变换。也就是从一个状态变化为另一个状态。例如“开灯过程”就是一个变换。

状态机的应用

状态机是一个对真实世界的抽象,而且是逻辑严谨的数学抽象,所以明显非常适合用在数字领域。可以应用到各个层面上,例如硬件设计,编译器设计,以及编程实现各种具体业务逻辑的时候。

进程5状态模型

进程管理是Linux五大子系统之一,非常重要,实际实现起来非常复杂,我们来看下进程是如何切换状态的。

下图是进程的5状态模型:

f4f1639e94af51def3a8a509f23ccd72.png

关于该图简单介绍如下:

  1. 可运行态:当进程正在被CPU执行,或已经准备就绪随时可由调度程序执行,则称该进程为处于运行状态(running)。进程可以在内核态运行,也可以在用户态运行。当系统资源已经可用时,进程就被唤醒而进入准备运行状态,该状态称为就绪态。
  2. 浅度睡眠态(可中断):进程正在睡眠(被阻塞),等待资源到来是唤醒,也可以通过其他进程信号或时钟中断唤醒,进入运行队列。
  3. 深度睡眠态(不可中断):其和浅度睡眠基本类似,但有一点就是不可由其他进程信号或时钟中断唤醒。只有被使用wake_up()函数明确唤醒时才能转换到可运行的就绪状态。
  4. 暂停状态:当进程收到信号SIGSTOP、SIGTSTP、SIGTTIN或SIGTTOU时就会进入暂停状态。可向其发送SIGCONT信号让进程转换到可运行状态。
  5. 僵死状态:当进程已停止运行,但其父进程还没有询问其状态时,未释放PCB,则称该进程处于僵死状态。

进程的状态就是按照这个状态图进行切换的。

该状态流程有点复杂,因为我们目标只是实现一个简单的状态机,所以我们简化一下该状态机如下:

f4f1639e94af51def3a8a509f23ccd72.png


要想实现状态机,首先将该状态机转换成下面的状态迁移表。

f4f1639e94af51def3a8a509f23ccd72.png

简要说明如下: 假设当前进程处于running状态下,那么只有schedule事件发生之后,该进程才会产生状态的迁移,迁移到owencpu状态下,如果在此状态下发生了其他的事件,比如wake、wait_event都不会导致状态的迁移。

如上图所示:

  1. 每一列表示一个状态,每一行对应一个事件。
  2. 该表是实现状态机的最核心的一个图,请读者详细对比该表和状态迁移图的的关系。
  3. 实际场景中,进程的切换会远比这个图复杂,好在众多大神都帮我们解决了这些复杂的问题,我们只需要站在巨人的肩膀上就可以了。

http://www.niftyadmin.cn/n/1485132.html

相关文章

java中如何自定义一个注解

1.创建一个 Annotation 类型的类 2.Target:指该注解的作用目标 3.Retention:指注解的保留策略 4.Inherited:指该注解是否可被继承

[转]提升“专注力”的10个策略

导读:看着案头上"未完成的事",为了逃避,我们会拖延和有意地将注意力分散到不重要的事情上去,那样做会给我们带来些许即时的满足感。等待我们的,却是深深的焦虑、自责。 一、我的故事 在我最近发现自己办事成…

yaml文件 .yml

YAML文件简介 我们可能在spring配置文件里见到过.yml格式的东东,配置文件不都是.propertie或者.xml文件吗?.yml是什么鬼,今天我带你们来一探究竟。 YAML(Yet Another Markup Language)(发音 /ˈjməl/ &…

python数据合并_如何使用python进行数据表的合并

案例背景假设我们的文件是放在G盘 python文件夹下单的projectFile文件夹中,具体的情况需根据读者文件位置进行设置 我们需要将下面两个文件,合并在一起Python学习交流群:1004391443 合并前,data_1.csv的数据合并前,dat…

idea 启动web项目报错 Can't load IA 64-bit .dll on a AMD 64-bit platform

网上解决办法基本一致认为: 如果上述办法没有解决你的问题,不妨试试下面这个办法。 -Xms512m -Xmx512m -XX:MaxNewSize1024m -XX:MaxPermSize1024m

cesium鼠标左键获取经纬度_Qt编写地图综合应用17-地址经纬度互转

一、前言 地址和经纬度互相转换的功能也经常用到,比如上次的路线方案查询的功能,之前官网是提供了直接输入出发地点和目的地的中文汉字,就可以查询到最优的路线,后面只支持输入出发地点和目的地的经纬度坐标了&#x…

BDF开发框架的搭建(摘)

原文来自:http://blog.csdn.net/wang102010/article/details/7286087 了解更多可以访问:http://bsdn.org/projects/bdf 提前拿到了bdf的源代码.顺手搭建了下开发环境.由于想在上面做商业开发,所以不得不把它的源代码解压出来搭建开发环境. 设计考虑以下几…

c++ 读入和写入文件

读入 #include<ifstream>ifstream infile; infile.open(img_dir); while(getline(infile,tmp)){} 写入 #include <fstream>ofstream outfile; outfile.open(txt_dir) outfile<<123<<endl; outfile.close(); 转载于:https://www.cnblogs.com/ymjyqsx/p/…