91欧美超碰AV自拍|国产成年人性爱视频免费看|亚洲 日韩 欧美一厂二区入|人人看人人爽人人操aV|丝袜美腿视频一区二区在线看|人人操人人爽人人爱|婷婷五月天超碰|97色色欧美亚州A√|另类A√无码精品一级av|欧美特级日韩特级

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫(xiě)文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

使用函數(shù)指針的方法實(shí)現(xiàn)狀態(tài)機(jī)

GReq_mcu168 ? 來(lái)源:玩轉(zhuǎn)單片機(jī) ? 作者:玩轉(zhuǎn)單片機(jī) ? 2020-10-19 09:36 ? 次閱讀
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

之前寫(xiě)過(guò)一篇狀態(tài)機(jī)的實(shí)用文章,很多朋友說(shuō)有幾個(gè)地方有點(diǎn)難度不易理解,今天給大家換種簡(jiǎn)單寫(xiě)法,使用函數(shù)指針的方法實(shí)現(xiàn)狀態(tài)機(jī)。

狀態(tài)機(jī)簡(jiǎn)介

有限狀態(tài)機(jī)FSM是有限個(gè)狀態(tài)及在這些狀態(tài)之間的轉(zhuǎn)移和動(dòng)作等行為的數(shù)學(xué)模型,是一種邏輯單元內(nèi)部的高效編程方法,可以根據(jù)不同狀態(tài)或者消息類(lèi)型進(jìn)行相應(yīng)的處理邏輯,使得程序邏輯清晰易懂。

函數(shù)指針實(shí)現(xiàn)FSM

使用函數(shù)指針實(shí)現(xiàn)FSM可以分為3個(gè)步驟

建立相應(yīng)的狀態(tài)表和動(dòng)作查詢(xún)表

根據(jù)狀態(tài)表、事件、動(dòng)作表定位相應(yīng)的動(dòng)作處理函數(shù)

執(zhí)行完成后再進(jìn)行狀態(tài)的切換

代碼實(shí)現(xiàn)步驟

定義狀態(tài)數(shù)據(jù)的枚舉類(lèi)型

typedefenum{ state_1=1, state_2, state_3, state_4 }State;

定義事件的枚舉類(lèi)型

typedefenum{ event_1=1, event_2, event_3, event_4, event_5 }EventID;

定義狀態(tài)表的數(shù)據(jù)類(lèi)型

typedefstruct { intevent;//事件 intCurState;//當(dāng)前狀態(tài) void(*eventActFun)();//函數(shù)指針 intNextState;//下一個(gè)狀態(tài) }StateTable;

定義處理函數(shù)及建立狀態(tài)表

voidf121() { printf("thisisf121 "); } voidf221() { printf("thisisf221 "); } voidf321() { printf("thisisf321 "); } voidf122() { printf("thisisf122 "); } StateTablefTable[]= { //{到來(lái)的事件,當(dāng)前的狀態(tài),將要要執(zhí)行的函數(shù),下一個(gè)狀態(tài)} {event_1,state_1,f121,event_2}, {event_2,state_2,f221,event_3}, {event_3,state_3,f321,event_4}, {event_4,state_4,f122,event_1}, //addyourcodehere };

狀態(tài)機(jī)類(lèi)型,及狀態(tài)機(jī)接口函數(shù)

/*狀態(tài)機(jī)類(lèi)型*/ typedefstruct{ intcurState;//當(dāng)前狀態(tài) StateTable*stateTable;//狀態(tài)表 intsize;//表的項(xiàng)數(shù) }fsmType; /*狀態(tài)機(jī)注冊(cè),給它一個(gè)狀態(tài)表*/ voidfsmRegist(fsmType*pFsm,StateTable*pTable) { pFsm->stateTable=pTable; } /*狀態(tài)遷移*/ voidfsmStateTransfer(fsmType*pFsm,intstate) { pFsm->curState=state; } /*事件處理*/ voidfsmEventHandle(fsmType*pFsm,intevent) { StateTable*pActTable=pFsm->stateTable; void(*eventActFun)()=NULL;//函數(shù)指針初始化為空 intNextState; intCurState=pFsm->curState; intmaxNum=pFsm->size; intflag=0;//標(biāo)識(shí)是否滿(mǎn)足條件 /*獲取當(dāng)前動(dòng)作函數(shù)*/ for(inti=0;i

附代碼

代碼直接復(fù)制過(guò)去就行啦,本想打包的,太麻煩了。

測(cè)試程序

//編譯器:http://www.dooccn.com/cpp/ //來(lái)源:技術(shù)讓夢(mèng)想更偉大 //作者:李肖遙 #include typedefenum{ state_1=1, state_2, state_3, state_4 }State; typedefenum{ event_1=1, event_2, event_3, event_4, event_5 }EventID; typedefstruct{ intevent;//事件 intCurState;//當(dāng)前狀態(tài) void(*eventActFun)();//函數(shù)指針 intNextState;//下一個(gè)狀態(tài) }StateTable; voidf121() { printf("thisisf121 "); } voidf221() { printf("thisisf221 "); } voidf321() { printf("thisisf321 "); } voidf122() { printf("thisisf122 "); } StateTablefTable[]= { //{到來(lái)的事件,當(dāng)前的狀態(tài),將要要執(zhí)行的函數(shù),下一個(gè)狀態(tài)} {event_1,state_1,f121,event_2}, {event_2,state_2,f221,event_3}, {event_3,state_3,f321,event_4}, {event_4,state_4,f122,event_1}, //addyourcodehere }; /*狀態(tài)機(jī)類(lèi)型*/ typedefstruct{ intcurState;//當(dāng)前狀態(tài) StateTable*stateTable;//狀態(tài)表 intsize;//表的項(xiàng)數(shù) }fsmType; /*狀態(tài)機(jī)注冊(cè),給它一個(gè)狀態(tài)表*/ voidfsmRegist(fsmType*pFsm,StateTable*pTable) { pFsm->stateTable=pTable; } /*狀態(tài)遷移*/ voidfsmStateTransfer(fsmType*pFsm,intstate) { pFsm->curState=state; } /*事件處理*/ voidfsmEventHandle(fsmType*pFsm,intevent) { StateTable*pActTable=pFsm->stateTable; void(*eventActFun)()=NULL;//函數(shù)指針初始化為空 intNextState; intCurState=pFsm->curState; intmaxNum=pFsm->size; intflag=0;//標(biāo)識(shí)是否滿(mǎn)足條件 /*獲取當(dāng)前動(dòng)作函數(shù)*/ for(inti=0;i

編譯結(jié)果

總結(jié)

使用函數(shù)指針實(shí)現(xiàn)的FSM的過(guò)程還是比較費(fèi)時(shí)費(fèi)力的,但是這一切相對(duì)一大堆的if/else、switch/case來(lái)說(shuō)都是值得的,當(dāng)你的程序規(guī)模變得越來(lái)越大的時(shí)候,基于這種表結(jié)構(gòu)的狀態(tài)機(jī),維護(hù)程序起來(lái)會(huì)清晰很多。

原文標(biāo)題:【編程之美】函數(shù)指針?lè)椒▽?shí)現(xiàn)簡(jiǎn)單狀態(tài)機(jī)(附代碼)

文章出處:【微信公眾號(hào):玩轉(zhuǎn)單片機(jī)】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

責(zé)任編輯:haq

聲明:本文內(nèi)容及配圖由入駐作者撰寫(xiě)或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問(wèn)題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • 編程
    +關(guān)注

    關(guān)注

    90

    文章

    3716

    瀏覽量

    97169
  • 函數(shù)
    +關(guān)注

    關(guān)注

    3

    文章

    4417

    瀏覽量

    67499
  • 指針
    +關(guān)注

    關(guān)注

    1

    文章

    484

    瀏覽量

    71843

原文標(biāo)題:【編程之美】函數(shù)指針?lè)椒▽?shí)現(xiàn)簡(jiǎn)單狀態(tài)機(jī)(附代碼)

文章出處:【微信號(hào):mcu168,微信公眾號(hào):硬件攻城獅】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

    評(píng)論

    相關(guān)推薦
    熱點(diǎn)推薦

    指針函數(shù)詳解

    = a + b; return(p); } 這是一個(gè)簡(jiǎn)單的指針函數(shù)的例子,運(yùn)行結(jié)果如下,本文代碼在VScode平臺(tái)運(yùn)行,使用方法《使用VScode調(diào)試C語(yǔ)言》。 不過(guò)我有個(gè)疑問(wèn),使用
    發(fā)表于 01-23 06:02

    函數(shù)指針介紹

    函數(shù)指針類(lèi)型 定義函數(shù)指針類(lèi)型,必須使用typedef,方法就是,在“定義函數(shù)
    發(fā)表于 01-21 08:11

    函數(shù)指針與回調(diào)函數(shù)解讀

    函數(shù)指針是指向函數(shù)指針變量。通過(guò)函數(shù)指針C語(yǔ)言可以實(shí)現(xiàn)
    發(fā)表于 01-19 07:34

    函數(shù)指針指針函數(shù)的區(qū)別

    = fun(a);   注意指針函數(shù)函數(shù)指針表示方法的不同,千萬(wàn)不要混淆。最簡(jiǎn)單的辨別方式就是看函數(shù)
    發(fā)表于 12-12 06:34

    函數(shù)指針的概念

    函數(shù)指針是指向函數(shù)指針變量。 通常我們說(shuō)的指針變量是指向一個(gè)整型、字符型或數(shù)組等變量,而函數(shù)
    發(fā)表于 12-11 08:10

    回調(diào)函數(shù)例子的應(yīng)用

    步驟,這里的的例子就是,利用一個(gè)狀態(tài)機(jī)函數(shù)(根據(jù)不同狀態(tài)依次調(diào)用不同實(shí)現(xiàn)方法函數(shù)),通過(guò)回調(diào)
    發(fā)表于 12-11 07:23

    如何用函數(shù)指針調(diào)用函數(shù)

    給大家舉一個(gè)例子: int Func(int x);/*聲明一個(gè)函數(shù)*/ int (*p) (int x);/*定義一個(gè)函數(shù)指針*/ p = Func; /*將Func函數(shù)的首地
    發(fā)表于 12-11 06:26

    條件判斷法來(lái)實(shí)現(xiàn)狀態(tài)機(jī)

    狀態(tài)用 switch—case 組織起來(lái), 將事件也用switch—case 組織起來(lái), 然后讓其中一個(gè) switch—case 整體插入到另一個(gè) switch—case 的每一個(gè) case 項(xiàng)中
    發(fā)表于 12-09 08:18

    睿遠(yuǎn)研究院丨IO-Link規(guī)范解讀(十一):ISDU狀態(tài)機(jī)與EVENT事件

    上篇我們介紹了ISDU的典型編碼格式和應(yīng)用案例,本篇我們就來(lái)詳細(xì)介紹下,ISDU的狀態(tài)機(jī),并把EVENT事件的邏輯,給大家好好解析下。 1主站ISDU狀態(tài)機(jī) 如上圖所示,ISDU的狀態(tài)機(jī)的核心
    的頭像 發(fā)表于 11-29 18:28 ?4714次閱讀
    睿遠(yuǎn)研究院丨IO-Link規(guī)范解讀(十一):ISDU<b class='flag-5'>狀態(tài)機(jī)</b>與EVENT事件

    什么是狀態(tài)機(jī)?

    的switch—case語(yǔ)句里要有400個(gè)case,這樣的程序還有法兒寫(xiě)么?! 同樣的功能改動(dòng),如果用g_stFSM這個(gè)結(jié)構(gòu)體來(lái)實(shí)現(xiàn)狀態(tài)機(jī)的話(huà),函數(shù)fsm_active()只需要
    發(fā)表于 11-27 08:15

    睿遠(yuǎn)研究院丨IO-Link規(guī)范解讀(六):主從站狀態(tài)機(jī)解析

    前言 書(shū)接上文,今天我們就來(lái)好好聊聊主從站的DL-Mode狀態(tài)機(jī),還請(qǐng)各位童鞋前排坐好! 1主站狀態(tài)機(jī)解析 主站的DL-Mode狀態(tài)機(jī)有5個(gè)大狀態(tài),也是我們很熟悉的 建立通信、開(kāi)始、預(yù)
    的頭像 發(fā)表于 10-28 17:34 ?6198次閱讀
    睿遠(yuǎn)研究院丨IO-Link規(guī)范解讀(六):主從站<b class='flag-5'>狀態(tài)機(jī)</b>解析

    通過(guò) BOD 或 nReset 重置時(shí),GPIO 是否處于高實(shí)現(xiàn)狀態(tài)

    正如標(biāo)題所說(shuō),我正在使用 ML51 來(lái)控制外部 MOS 組件,GPIO 類(lèi)型在復(fù)位條件下非常重要。GPIO 是否處于高實(shí)現(xiàn)狀態(tài)?
    發(fā)表于 08-25 07:04

    有可能在 FX3 GPIF2 中創(chuàng)建兩個(gè)獨(dú)立的狀態(tài)機(jī)嗎?

    我想,如果我想通過(guò) FX3 GPIF2 創(chuàng)建兩個(gè)獨(dú)立的傳輸流接口,我需要在 GPIF2 設(shè)計(jì)器中創(chuàng)建兩個(gè)獨(dú)立的狀態(tài)機(jī),我是否有可能在 GPIF2 設(shè)計(jì)器中創(chuàng)建兩個(gè)獨(dú)立的狀態(tài)機(jī)?
    發(fā)表于 05-20 06:14

    求助,關(guān)于srammaster.cydsn中狀態(tài)機(jī)的問(wèn)題求解

    晚上好。 我目前正在學(xué)習(xí) GPIF II。 查看..EZ-USB FX3 SDK1.3firmwaregpif_examplescyfxsrammastersrammaster.cydsn中的狀態(tài)機(jī),有狀態(tài)START和START1。 這意味著什么?
    發(fā)表于 05-12 06:20

    函數(shù)指針的六個(gè)常見(jiàn)應(yīng)用場(chǎng)景

    函數(shù)指針在嵌入式開(kāi)發(fā)中有著廣泛的應(yīng)用,它讓代碼更加靈活,減少冗余,提高可擴(kuò)展性。很多時(shí)候,我們需要根據(jù)不同的情況動(dòng)態(tài)調(diào)用不同的函數(shù),而函數(shù)指針
    的頭像 發(fā)表于 04-07 11:58 ?1471次閱讀
    <b class='flag-5'>函數(shù)</b><b class='flag-5'>指針</b>的六個(gè)常見(jiàn)應(yīng)用場(chǎng)景