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

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

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

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

GReq_mcu168 ? 來源:玩轉單片機 ? 作者:玩轉單片機 ? 2020-10-19 09:36 ? 次閱讀

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

狀態(tài)機簡介

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

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

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

建立相應的狀態(tài)表和動作查詢表

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

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

代碼實現(xiàn)步驟

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

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

定義事件的枚舉類型

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

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

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

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

voidf121() { printf("thisisf121 "); } voidf221() { printf("thisisf221 "); } voidf321() { printf("thisisf321 "); } voidf122() { printf("thisisf122 "); } StateTablefTable[]= { //{到來的事件,當前的狀態(tài),將要要執(zhí)行的函數(shù),下一個狀態(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)機類型,及狀態(tài)機接口函數(shù)

/*狀態(tài)機類型*/ typedefstruct{ intcurState;//當前狀態(tài) StateTable*stateTable;//狀態(tài)表 intsize;//表的項數(shù) }fsmType; /*狀態(tài)機注冊,給它一個狀態(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;//標識是否滿足條件 /*獲取當前動作函數(shù)*/ for(inti=0;i

附代碼

代碼直接復制過去就行啦,本想打包的,太麻煩了。

測試程序

//編譯器:http://www.dooccn.com/cpp/ //來源:技術讓夢想更偉大 //作者:李肖遙 #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;//當前狀態(tài) void(*eventActFun)();//函數(shù)指針 intNextState;//下一個狀態(tài) }StateTable; voidf121() { printf("thisisf121 "); } voidf221() { printf("thisisf221 "); } voidf321() { printf("thisisf321 "); } voidf122() { printf("thisisf122 "); } StateTablefTable[]= { //{到來的事件,當前的狀態(tài),將要要執(zhí)行的函數(shù),下一個狀態(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)機類型*/ typedefstruct{ intcurState;//當前狀態(tài) StateTable*stateTable;//狀態(tài)表 intsize;//表的項數(shù) }fsmType; /*狀態(tài)機注冊,給它一個狀態(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;//標識是否滿足條件 /*獲取當前動作函數(shù)*/ for(inti=0;i

編譯結果

總結

使用函數(shù)指針實現(xiàn)的FSM的過程還是比較費時費力的,但是這一切相對一大堆的if/else、switch/case來說都是值得的,當你的程序規(guī)模變得越來越大的時候,基于這種表結構的狀態(tài)機,維護程序起來會清晰很多。

原文標題:【編程之美】函數(shù)指針方法實現(xiàn)簡單狀態(tài)機(附代碼)

文章出處:【微信公眾號:玩轉單片機】歡迎添加關注!文章轉載請注明出處。

責任編輯:haq

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

    關注

    88

    文章

    3674

    瀏覽量

    94752
  • 函數(shù)
    +關注

    關注

    3

    文章

    4365

    瀏覽量

    63959
  • 指針
    +關注

    關注

    1

    文章

    484

    瀏覽量

    70946

原文標題:【編程之美】函數(shù)指針方法實現(xiàn)簡單狀態(tài)機(附代碼)

文章出處:【微信號:mcu168,微信公眾號:硬件攻城獅】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦
    熱點推薦

    求助,關于srammaster.cydsn中狀態(tài)機的問題求解

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

    NVME控制器之隊列管理模塊

    時,表示隊列為滿;當Head指針等于Tail指針時,表示隊列為空。該模塊中的狀態(tài)機用來實現(xiàn)門鈴寄存器信息更新的流程控制工作。隊列管理狀態(tài)機
    發(fā)表于 05-03 20:19

    函數(shù)指針的六個常見應用場景

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

    CSU-IDE是否支持函數(shù)指針

    在編寫代碼過程中需要使用函數(shù)指針,編譯的時候報不支持,請問是需要設置,還是軟件本身就不支持
    發(fā)表于 12-22 23:02

    Simulink中的狀態(tài)機建模方法 Simulink數(shù)據(jù)可視化與分析功能

    1. Simulink中的狀態(tài)機建模方法 1.1 理解狀態(tài)機的基本概念 在開始建模之前,了解狀態(tài)機的基本概念是必要的。狀態(tài)機由以下幾個部分組
    的頭像 發(fā)表于 12-12 09:27 ?2377次閱讀

    C語言指針詳細解析

    函數(shù)入?yún)⒍际窍嗤模@時候如果要調(diào)用不同的排序方法,就可以使用指針函數(shù)實現(xiàn),我們只需要修改函數(shù)
    發(fā)表于 09-14 10:03

    觸發(fā)器和狀態(tài)機的關系是什么

    觸發(fā)器和狀態(tài)機在數(shù)字電路設計中有著緊密的關系,它們共同構成了時序邏輯電路的基礎,用于實現(xiàn)數(shù)據(jù)的存儲、處理和傳輸。
    的頭像 發(fā)表于 08-12 11:24 ?768次閱讀

    面試???1:函數(shù)指針指針函數(shù)、數(shù)組指針指針數(shù)組

    在嵌入式開發(fā)領域,函數(shù)指針、指針函數(shù)、數(shù)組指針指針數(shù)組是一些非常重要但又容易混淆的概念。理解它
    的頭像 發(fā)表于 08-10 08:11 ?1243次閱讀
    面試常考+1:<b class='flag-5'>函數(shù)</b><b class='flag-5'>指針</b>與<b class='flag-5'>指針</b><b class='flag-5'>函數(shù)</b>、數(shù)組<b class='flag-5'>指針</b>與<b class='flag-5'>指針</b>數(shù)組

    如何在FPGA中實現(xiàn)狀態(tài)機

    在FPGA(現(xiàn)場可編程門陣列)中實現(xiàn)狀態(tài)機是一種常見的做法,用于控制復雜的數(shù)字系統(tǒng)行為。狀態(tài)機能夠根據(jù)當前的輸入和系統(tǒng)狀態(tài),決定下一步的動作和新的狀態(tài)。這里,我們將詳細探討如何在FPG
    的頭像 發(fā)表于 07-18 15:57 ?1021次閱讀

    玩轉Spring狀態(tài)機

    說起Spring狀態(tài)機,大家很容易聯(lián)想到這個狀態(tài)機和設計模式中狀態(tài)模式的區(qū)別是啥呢?沒錯,Spring狀態(tài)機就是狀態(tài)模式的一種
    的頭像 發(fā)表于 06-25 14:21 ?1225次閱讀
    玩轉Spring<b class='flag-5'>狀態(tài)機</b>

    面試中的高頻問題:指針函數(shù)函數(shù)指針,你能完美應對嗎?

    一直覺得C語言較其他語言最偉大的地方就是C語言中的指針,有些人認為指針很簡單,而有些人認為指針很難,當然這里的對簡單和難并不是等價于對指針的理解程度。為此在這里對C語言中的
    的頭像 發(fā)表于 06-22 08:11 ?2143次閱讀
    面試中的高頻問題:<b class='flag-5'>指針</b><b class='flag-5'>函數(shù)</b>與<b class='flag-5'>函數(shù)</b><b class='flag-5'>指針</b>,你能完美應對嗎?

    上位如何實時讀plc的狀態(tài)

    讀取PLC狀態(tài)方法,包括通信協(xié)議、硬件連接、軟件編程等方面的內(nèi)容。 1. 通信協(xié)議 在實現(xiàn)上位與PLC之間的通信時,需要選擇合適的通信協(xié)議。常見的通信協(xié)議有Modbus、Profi
    的頭像 發(fā)表于 06-06 10:09 ?2852次閱讀

    關于SMU狀態(tài)機的問題求解

    我有一些關于 SMU 狀態(tài)機的問題。 假設由于某種原因,SMU 已進入故障狀態(tài)。 手冊指出,要返回運行狀態(tài)并將 FSP 恢復到無故障狀態(tài),應調(diào)用IfxSmu_releaseFSP()。
    發(fā)表于 05-29 08:18

    指針式萬用表的讀數(shù)方法

    指針式萬用表作為電子測量中常用的工具,其準確性和便利性受到了廣泛的認可。然而,對于初學者來說,如何正確讀取指針式萬用表的示數(shù)卻是一個需要掌握的技能。本文將詳細介紹指針式萬用表的讀數(shù)方法
    的頭像 發(fā)表于 05-20 17:12 ?3266次閱讀

    使用系統(tǒng)滴答定時中斷,基于按鍵的狀態(tài)機怎么只能1個1個+,不能連+?

    使用系統(tǒng)滴答定時中斷,基于按鍵的狀態(tài)機怎么只能1個1個+,不能連+ #define KEY1_USERGPIO_ReadInputDataBit(GPIOC,GPIO_Pin_13
    發(fā)表于 05-16 06:27

    電子發(fā)燒友

    中國電子工程師最喜歡的網(wǎng)站

    • 2931785位工程師會員交流學習
    • 獲取您個性化的科技前沿技術信息
    • 參加活動獲取豐厚的禮品