在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

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

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

3天內不再提示

C語言實現狀態機的主要3種方法

STM32嵌入式開發 ? 來源:STM32嵌入式開發 ? 2023-11-06 12:20 ? 次閱讀

狀態機的實現無非就是 3 個要素:狀態、事件、響應。轉換成具體的行為就 3 句話。

發生了什么事?

現在系統處在什么狀態?

在這樣的狀態下發生了這樣的事,系統要干什么?

用 C 語言實現狀態機主要有 3 種方法:switch—case 法、表格驅動法、函數指針法。

switch—case 法

狀態用 switch—case 組織起來, 將事件也用switch—case 組織起來, 然后讓其中一個 switch—case 整體插入到另一個 switch—case 的每一個 case 項中 。 「程序清單 List4 :」


switch(StateVal)
{
    case S0:
  switch(EvntID)
  {
   case E1:
    action_S0_E1(); /*S0 狀態下 E1 事件的響應*/
    StateVal = new state value;/*狀態遷移,不遷移則沒有此行*/
    break;
   case E2:
    action_S0_E2(); /*S0 狀態下 E2 事件的響應*/
    StateVal = new state value;
    break;
   ......
   case Em:
    action_S0_Em(); /*S0 狀態下 Em 事件的響應*/
    StateVal = new state value;
    break;
   default:
    break;
  }
  break;
    case S1:
  ......
  break;
    ......
    case Sn:
  ......
  break;
    default:
  break;
}
上面的偽代碼示例只是通用的情況,實際應用遠沒有這么復雜。雖然一個系統中事件可能有很多種,但在實際應用中,許多事件可能對某個狀態是沒有意義的。 例如在程序清單 List4中,如果 E2、······ Em 對處在 S0 狀態下的系統沒有意義,那么在 S0 的 case 下有關事件E2、······ Em 的代碼根本沒有必要寫,狀態 S0 只需要考慮事件 E1 的處理就行了。 既然是兩個 switch—case 之間的嵌套, 那么就有一個誰嵌套誰的問題, 所以說 switch—case法有兩種寫法:狀態嵌套事件和事件嵌套狀態。這兩種寫法都可以, 各有利弊, 至于到底選用哪種方式就留給設計人員根據具體情況自行決斷吧。 關于 switch—case 法還有最后一點要說明, 因為 switch—case 的原理是從上到下挨個比較,越靠后,查找耗費的時間就越長,所以要注意狀態和事件在各自的 switch 語句中的安排順序,不推薦程序清單 List4 那樣按順序號排布的方式。出現頻率高或者實時性要求高的狀態和事件的位置應該盡量靠前。

表格驅動法

如果說 switch—case 法是線性的,那么表格驅動法則是平面的。表格驅動法的實質就是將狀態和事件之間的關系固化到一張二維表格里, 把事件當做縱軸,把狀態當做橫軸,交點[Sn , Em]則是系統在 Sn 狀態下對事件 Em 的響應 。 c07aa454-7c5a-11ee-939d-92fbcf53809c.png 如圖 4, 我把表格中的 Node_SnEm 叫做狀態機節點, 狀態機節點 Node_SnEm 是系統在 Sn狀態下對事件 Em 的響應。這里所說的響應包含兩個方面:輸出動作和狀態遷移。狀態機節點一般是一個類似程序清單 List5 中的結構體變量 。


struct fsm_node
{
    void (*fpAction)(void* pEvnt);
    INT8U u8NxtStat;
};
程序清單 List5 中的這個結構體有兩個成員:fpAction 和 u8NxtStat。fpAction 是一個函數指針, 指向一個形式為 void func(void * pEvnt)的函數, func 這個函數是對狀態轉移中動作序列的標準化封裝。 也就是說, 狀態機在狀態遷移的時候, 不管輸出多少個動作、操作多少個變量、調用多少個函數,這些行為統統放到函數 func 中去做。 把動作封裝好了之后,再把封裝函數 func 的地址交給函數指針 fpAction,這樣,想要輸出動作,只需要調用函數指針 fpAction 就行了。 再看看上面的 func 函數,會發現函數有一個形參 pEvnt,這是一個類型為 void * 的指針, 在程序實際運行時指向一個能存儲事件的變量,通過這個指針我們就能獲知關于事件的全部信息,這個形參是很有必要的。事件一般包括兩個屬性:事件的類型和事件的內容。 例如一次按鍵事件,我們不僅要知道這是一個按鍵事件,還要知道按下的到底是哪個鍵。事件的類型和狀態機當前的狀態可以讓我們在圖 4 的表格中迅速定位,確定該調用哪個動作封裝函數, 但是動作封裝函數要正確響應事件還需要知道事件的內容是什么, 這也就是形參pEvnt 的意義。 由于事件的多樣性,存儲事件內容的數據格式不一定一樣,所以就把 pEvnt 定義成了 void * 型,以增加靈活性。有關 fpAction 的最后一個問題:如果事件 Em 對狀態 Sn 沒有意義,那么狀態機節點Node_SnEm 中的 fpAction 該怎么辦?我的答案是:那就讓它指向一個空函數唄!前面不是說過么,什么也不干也叫響應。 u8NxtStat 存儲的是狀態機的一個狀態值。我們知道, 狀態機響應事件要輸出動作, 也就是調用函數指針 fpAction 所指向的那個封裝函數, 函數調用完畢后程序返回主調函數, 狀態機對事件的響應就算結束了, 下一步就要考慮狀態遷移的問題了。 可能要保持本狀態不變, 也可能要遷移到一個新的狀態,該如何抉擇呢?u8NxtStat 存儲的狀態就是狀態機想要的答案! 圖 4 的這張表格反映在 C 語言代碼里就是一個二維數組,第 1 維就是狀態機的狀態,第 2維就是統一分類的事件,而數組的元素則是程序清單 List5 中的結構體常量。如果程序中使用表格驅動法,還需要注意一些特別的事項。要將狀態當做表格的橫軸,那么就要求狀態值集合必須滿足以下條件: (1) 該集合是一個遞增的等差整數數列 (2) 該數列初值為 0 (3) 該數列等差值為 1 “事件” 作為縱軸,其特點和要求與用來做橫軸的“狀態” 完全一致。在 C 語言提供的數據類型中, 沒有比枚舉更符合以上要求的可選項了, 極力推薦將狀態集合和事件類型集合做成枚舉常量。表格驅動法的優點:調用接口統一 ,定位快速。 表格驅動法屏蔽了不同狀態下處理各個事件的差異性,因此可以將處理過程中的共性部分提煉出來,做成標準統一的框架式代碼,形成統一的調用接口。根據程序清單 List5 中的狀態機節點結構體,做成的框架代碼如程序清單 List6 所示。 表格驅動法查找目標實際上就是一次二維數組的尋址操作,所以它的平均效率要遠高于switch—case 法。 「程序清單 List6 :」

extern struct fsm_node g_arFsmDrvTbl[][]; /*狀態機驅動表格*/
INT8U u8CurStat = 0; /*狀態暫存*/
INT8U u8EvntTyp = 0; /*事件類型暫存*/
void* pEvnt = NULL; /*事件變量地址暫存*/
struct fsm_node stNodeTmp = {NULL, 0}; /*狀態機節點暫存*/
u8CurStat = get_cur_state(); /*讀取當前狀態*/
u8EvntTyp = get_cur_evnt_typ(); /*讀取當前觸發事件類型*/
pEvnt = (void*)get_cur_evnt_ptr(); /*讀取事件變量地址*/
stNodeTmp = g_arFsmDrvTbl[u8CurStat ][u8EvntTyp ];/*定位狀態機節點*/
stNodeTmp.fpAction(pEvnt ); /*動作響應*/
set_cur_state(stNodeTmp.u8NxtStat); /*狀態遷移*/
.....
表格驅動法好則好矣,但用它寫出來的程序還有點兒小問題,我們先來看看按照表格驅動法寫出來的程序有什么特點 。 前面說過,表格驅動法可以把狀態機調度的部分做成標準統一的框架代碼,這個框架適用性極強, 不管用狀態機來實現什么樣的應用, 框架代碼都不需要做改動, 我們只需要根據實際應用場合規劃好狀態轉換圖,然后將圖中的各個要素(狀態、事件、動作、遷移,有關“條件”要素一會兒再說)用代碼實現就行了,我把這部分代碼稱作應用代碼。 在應用代碼的.c 文件中, 你會看到一個聲明為 const 的二維數組, 也就是圖 4 所示的狀態驅動表格, 還會看到許多彼此之間毫無關聯的函數, 也就是前面提到的動作封裝函數。這樣的一份代碼, 如果手頭上沒有一張狀態轉換圖, 讓誰看了也會一頭霧水, 這樣的格式直接帶來了代碼可讀性差的問題。 如果我們想給狀態機再添加一個狀態,反映到代碼上就是給驅動表格再加一列內容,同時也要新添加若干個動作封裝函數。如果驅動表格很大, 做這些工作是很費事兒的, 而且容易出錯。如果不小心在數組中填錯了位置, 那么程序跑起來就和設計者的意圖南轅北轍了, 遠沒有在 switch—case 法中改動來得方便、安全。Extended State Machine 的最大特點就是狀態機響應事件之前先判斷條件,根據判定結果選擇執行哪些動作,轉向哪個狀態。 也就是說,系統在狀態 Sn 下發生了事件 Em 后,轉向的狀態不一定是唯一的,這種靈活性是 Extended State Machine 的最有價值的優點。 回過頭來看看程序清單 List5 中給出的狀態機節點結構體,如果系統在狀態 Sn 下發生了事件 Em, 狀態機執行完 fpAction 所給出的動作響應之后, 必須轉到 u8NxtStat 指定的狀態。 表格驅動法的這個特性直接杜絕了 Extended State Machine 在表格驅動法中應用的可能性, 所以表格驅動法的代碼實現中不存在“條件” 這個狀態機要素。ESM,你是如此的優秀,我怎么舍得拋棄你 ?! 再看圖 4 所示的表格驅動法示例圖,如果我們把表格中的代表事件的縱軸去掉,只留下代表狀態的橫軸,將一列合并成一格,前文提到的問題是不是能得到解決呢?不錯!這就是失傳江湖多年的《葵花寶典》 ——閹割版表格驅動法 !! 閹割版表格驅動法,又名壓縮表格驅動法,一維狀態表格與事件 switch—case 的合體。壓縮表格驅動法使用了一維數組作為驅動表格,數組的下標即是狀態機的各個狀態。 表格中的元素叫做壓縮狀態機節點, 節點的主要內容還是一個指向動作封裝函數的函數指針, 只不過這個動作封裝函數不是為某個特定事件準備的, 而是對所有的事件都有效的。 節點中不再強制指定狀態機輸出動作完畢后所轉向的狀態, 而是讓動作封裝函數返回一個狀態, 并把這個狀態作為狀態機新的狀態。 壓縮表格驅動法的這個特點, 完美的解決了 Extended State Machine 不能在表格驅動法中使用的問題 。 程序清單 List7 中的示例代碼包含了壓縮狀態機節點結構體和狀態機調用的框架代碼。 「程序清單 List7:」

struct fsm_node /*壓縮狀態機節點結構體*/
{
 INT8U (*fpAction)(void* pEvnt); /*事件處理函數指針*/
 INT8U u8StatChk; /*狀態校驗*/
};
......
u8CurStat = get_cur_state(); /*讀取當前狀態*/
......
if(stNodeTmp.u8StatChk == u8CurStat )
{
 u8CurStat = stNodeTmp.fpAction(pEvnt ); /*事件處理*/
 set_cur_state(u8CurStat ); /*狀態遷移*/
}
else
{
 state_crash(u8CurStat ); /*非法狀態處理*/
}
.....
對照程序清單 List5,就會發現程序清單 List7 中 struct fsm_node 結構體的改動之處。首先, fpAction 所指向函數的函數形式變了,動作封裝函數 func 的模樣成了這樣的了:

INT8U func(void * pEvnt);
現在的動作封裝函數 func 是要返回類型為 INT8U 的返回值的,這個返回值就是狀態機要轉向的狀態, 也就是說, 壓縮表格驅動法中的狀態機節點不負責狀態機新狀態的確定, 而把這項任務交給了動作封裝函數 func, func 返回哪個狀態, 狀態機就轉向哪個狀態。 新狀態由原來的常量變成了現在的變量,自然要靈活許多。上面說到現在的動作封裝函數 func 要對當前發生的所有的事件都要負責, 那么 func 怎么會知道到底是哪個事件觸發了它呢?看一下 func 的形參 void * pEvnt 。 在程序清單 List5 中我們提到過,這個形參是用來向動作封裝函數傳遞事件內容的,但是從前文的敘述中我們知道, pEvnt 所指向的內存包含了事件的所有信息, 包括事件類型和事件內容 , 所以通過形參 pEvnt , 動作封裝函數 func 照樣可以知道事件的類型。 程序清單 List7 中 struct fsm_node 結構體還有一個成員 u8StatChk , 這里面存儲的是狀態機 的一個狀態,干什么用的呢?玩 C 語言數組的人都知道,要嚴防數組尋址越界。 要知道,壓縮表格驅動法的驅動表格是一個以狀態值為下標的一維數組, 數組元素里面最重要的部分就是一個個動作封裝函數的地址。 函數地址在單片機看來無非就是一段二進制數據, 和內存中其它的二進制數據沒什么兩樣,不管程序往單片機 PC 寄存器里塞什么值,單片機都沒意見。假設程序由于某種意外而改動了存儲狀態機當前狀態的變量,使變量值變成了一個非法狀態。 再發生事件時, 程序就會用這個非法的狀態值在驅動表格中尋址, 這時候就會發生內存泄露,程序拿泄露內存中的未知數據當函數地址跳轉,不跑飛才怪! 為了防止這種現象的發生, 壓縮狀態機節點結構體中又添加了成員 u8StatChk 。u8StatChk中存儲的是壓縮狀態機節點在一維驅動表格的位置, 例如某節點是表格中的第 7 個元素, 那么這個節點的成員 u8StatChk 值就是 6。 看一下程序清單 List7 中的框架代碼示例, 程序在引用函數指針 fpAction 之前, 先檢查當前狀態和當前節點成員 u8CurStat 的值是否一致,一致則認為狀態合法,事件正常響應,如果不一致,則認為當前狀態非法,轉至意外處理,最大限度保證程序運行的安全。當然,如果泄露內存中的數據恰好和 u8CurStat 一致,那么這種方法真的就回天乏力了。 還有一個方法也可以防止狀態機跑飛,如果狀態變量是枚舉,那么框架代碼就可以獲知狀態值的最大值, 在調用動作封裝函數之前判斷一下當前狀態值是否在合法的范圍之內, 同樣能保證狀態機的安全運行。 壓縮表格驅動法中動作封裝函數的定義形式我們已經知道了,函數里面到底是什么樣子的呢?程序清單 List8 是一個標準的示例。 「程序清單List8:」

INT8U action_S0(void* pEvnt)
{
 INT8U u8NxtStat = 0;
 INT8U u8EvntTyp = get_evnt_typ(pEvnt);
 switch(u8EvntTyp )
 {
  case E1:
   action_S0_E1(); /*事件 E1 的動作響應*/
   u8NxtStat = new state value; /*狀態遷移,不遷移也必須有本行*/
   break;
   ......
  case Em:
   action_S0_Em(); /*事件 Em 的動作響應*/
   u8NxtStat = new state value; /*狀態遷移,不遷移也必須有本行*/
   break;
  default:
   ; /*不相關事件處理*/
   break;
 }
 return u8NxtStat ; /*返回新狀態*/
}
從程序清單 List8 可以看出, 動作封裝函數其實就是事件 switch—case 的具體實現。函數根據形參 pEvnt 獲知事件類型, 并根據事件類型選擇動作響應, 確定狀態機遷移狀態, 最后將新的狀態作為執行結果返回給框架代碼。 有了這樣的動作封裝函數, Extended State Machine 的應用就可以完全不受限制了!到此,有關壓縮表格驅動法的介紹就結束了。 個人認為壓縮表格驅動法是相當優秀的,它既有表格驅動法的簡潔、高效、標準,又有 switch—case 法的直白、靈活、多變,相互取長補短,相得益彰。

函數指針法

上面說過,用 C 語言實現狀態機主要有 3 種方法(switch—case 法、表格驅動法、函數指針法), 其中函數指針法是最難理解的, 它的實質就是把動作封裝函數的函數地址作為狀態來看待。不過,有了之前壓縮表格驅動法的鋪墊,函數指針法就變得好理解了,因為兩者本質上是相同的。 壓縮表格驅動法的實質就是一個整數值(狀態機的一個狀態)到一個函數地址(動作封裝函數)的一對一映射, 壓縮表格驅動法的驅動表格就是全部映射關系的直接載體。在驅動表格中通過狀態值就能找到函數地址,通過函數地址同樣能反向找到狀態值。 我們用一個全局的整型變量來記錄狀態值,然后再查驅動表格找函數地址,那干脆直接用一個全局的函數指針來記錄狀態得了,還費那勞什子勁干嗎?!這就是函數指針法的前世今生。 用函數指針法寫出來的動作封裝函數和程序清單 List8 的示例函數是很相近的, 只不過函數的返回值不再是整型的狀態值, 而是下一個動作封裝函數的函數地址, 函數返回后, 框架代碼再把這個函數地址存儲到全局函數指針變量中。 相比壓縮表格驅動法,在函數指針法中狀態機的安全運行是個大問題,我們很難找出一種機制來檢查全局函數指針變量中的函數地址是不是合法值。如果放任不管, 一旦函數指針變量中的數據被篡改,程序跑飛幾乎就不可避免了。

總結

有關狀態機的東西說了那么多,相信大家都已經感受到了這種工具的優越性,狀態機真的是太好用了!其實我們至始至終講的都是有限狀態機(Finite State Machine 現在知道為什么前面的代碼中老是有 fsm 這個縮寫了吧!), 還有一種比有限狀態機更 NB 更復雜的狀態機, 那就是層次狀態機(Hierarchical State Machine 一般簡寫為 HSM)。 通俗的說,系統中只存在一個狀態機的叫做有限狀態機,同時存在多個狀態機的叫做層次狀態機(其實這樣解釋層次狀態機有些不嚴謹, 并行狀態機也有多個狀態機, 但層次狀態機各個狀態機之間是上下級關系,而并行狀態機各個狀態機之間是平級關系)。 層次狀態機是一種父狀態機包含子狀態機的多狀態機結構,里面包含了許多與面向對象相似的思想, 所以它的功能也要比有限狀態機更加強大, 當一個問題用有限狀態機解決起來有些吃力的時候, 就需要層次狀態機出馬了。 層次狀態機理論我理解得也不透徹, 就不在這里班門弄斧了,大家可以找一些有關狀態機理論的專業書籍來讀一讀。要掌握狀態機編程,理解狀態機(主要指有限狀態機)只是第一步,也是最簡單的一步,更重要的技能是能用狀態機這個工具去分析解剖實際問題:劃分狀態、 提取事件、 確定轉換關系、規定動作等等,形成一張完整的狀態轉換圖,最后還要對轉換圖進行優化,達到最佳。 把實際問題變成了狀態轉換圖, 工作的一大半就算完成了, 這個是具有架構師氣質的任務,剩下的問題就是按照狀態圖編程寫代碼了,這個是具有代碼工特色的工作。

編輯:黃飛

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

    關注

    1

    文章

    533

    瀏覽量

    58297
  • C語言
    +關注

    關注

    180

    文章

    7608

    瀏覽量

    137130
  • 函數
    +關注

    關注

    3

    文章

    4338

    瀏覽量

    62749
  • 指針
    +關注

    關注

    1

    文章

    480

    瀏覽量

    70582
  • 狀態機
    +關注

    關注

    2

    文章

    492

    瀏覽量

    27577

原文標題:C語言實現狀態機的三種方法

文章出處:【微信號:c-stm32,微信公眾號:STM32嵌入式開發】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    C語言實現狀態機設計模式

    狀態機模式是一行為模式,在《設計模式》這本書中對其有詳細的描述,通過多態實現不同狀態的調轉行為的確是一很好的
    發表于 12-14 13:38 ?2841次閱讀

    狀態機編程實例-面向對象的狀態設計模式

    本編介紹了狀態機編程的第3種方法——面向對象的狀態設計模式,通過C++的繼承特性,以及類指針,實現
    的頭像 發表于 06-28 09:04 ?1548次閱讀
    <b class='flag-5'>狀態機</b>編程實例-面向對象的<b class='flag-5'>狀態</b>設計模式

    C語言實現嵌入式狀態機的三種方法

    狀態機實現無非就是三個要素:狀態、事件、響應。
    發表于 07-02 11:00 ?1815次閱讀
    <b class='flag-5'>C</b><b class='flag-5'>語言實現</b>嵌入式<b class='flag-5'>狀態機</b>的三<b class='flag-5'>種方法</b>

    基于C語言狀態機實現方案

    關于狀態機,基礎的知識點可以自行理解。本文主要講解的是一個有限狀態機FSM通用的寫法,目的在于更好理解,移植,節省代碼閱讀與調試時間,體現出編程之美。
    發表于 09-13 09:28 ?873次閱讀
    基于<b class='flag-5'>C</b><b class='flag-5'>語言</b>的<b class='flag-5'>狀態機</b><b class='flag-5'>實現</b>方案

    如何在微型計算機中實現狀態機

    我不是C語言,我是為18F的微型計算機組裝的。我一直在考慮用微控制器來控制電梯。在閱讀了幾篇文章和參考文獻之后,我發現狀態機經常(如果不總是)被提及,我所理解的是考慮問題的方法。然后,
    發表于 09-30 09:18

    【工程源碼】基于FPGA在Modelsim仿真中顯示狀態機名稱的3種方法

    經過網上的搜索及自己的實驗,總結在Modelsim仿真中顯示狀態機名稱的三種方法。下面以一個具體的實例進行講解。實例功能:引入狀態機實現2分頻,這里使用
    發表于 02-21 16:48

    有限狀態機的硬件描述語言設計方法

    實驗目的 1、 熟悉用硬件描述語言(VHDL)設計一般狀態機所包含的幾個基本部分;2、 掌握用硬件描述語言(VHDL)設計Moore型和Mealy型有限狀態機
    發表于 09-03 09:48 ?0次下載

    基于FPGA實現狀態機的設計

    狀態機有三描述方式:一段式狀態機、兩段式狀態機、三段式狀態機。下面就用一個小例子來看看三方式
    的頭像 發表于 08-29 06:09 ?2865次閱讀
    基于FPGA<b class='flag-5'>實現狀態機</b>的設計

    什么是狀態機 狀態機的描述三種方法

    狀態機 1、狀態機是許多數字系統的核心部件,是一類重要的時序邏輯電路。通常包括三個部分:一是下一個狀態的邏輯電路,二是存儲狀態機當前狀態的時
    的頭像 發表于 11-16 17:39 ?2.7w次閱讀

    使用函數指針的方法實現狀態機

    之前寫過一篇狀態機的實用文章,很多朋友說有幾個地方有點難度不易理解,今天給大家換簡單寫法,使用函數指針的方法實現狀態機狀態機簡介 有限
    的頭像 發表于 10-19 09:36 ?2425次閱讀
    使用函數指針的<b class='flag-5'>方法</b><b class='flag-5'>實現狀態機</b>

    使用單片實現24C02存儲上次使用中狀態C語言實例免費下載

    本文檔的主要內容詳細介紹的是使用單片實現24C02存儲上次使用中狀態C
    發表于 03-24 13:51 ?13次下載

    單片實現24C02存儲上次使用中狀態C語言實

    單片實現24C02存儲上次使用中狀態C語言實
    發表于 03-15 14:19 ?11次下載

    如何在FPGA中實現狀態機

    狀態機往往是FPGA 開發的主力。選擇合適的架構和實現方法將確保您獲得一款最佳解決方案。 FPGA 常常用于執行基于序列和控制的行動, 比如實現一個簡單的通信協議。對于設計人員來說,滿
    的頭像 發表于 07-18 16:05 ?1116次閱讀
    如何在FPGA中<b class='flag-5'>實現狀態機</b>

    狀態機的三實現模式(C語言實現狀態機的三種方法

    壓縮表格驅動法的實質就是一個整數值(狀態機的一個狀態)到一個函數地址(動作封裝函數)的一對一映射, 壓縮表格驅動法的驅動表格就是全部映射關系的直接載體。在驅動表格中通過狀態值就能找到函數地址,通過函數地址同樣能反向找到
    發表于 07-25 10:27 ?3165次閱讀
    <b class='flag-5'>狀態機</b>的三<b class='flag-5'>種</b><b class='flag-5'>實現</b>模式(<b class='flag-5'>C</b><b class='flag-5'>語言實現狀態機</b>的三<b class='flag-5'>種方法</b>)

    如何在FPGA中實現狀態機

    在FPGA(現場可編程門陣列)中實現狀態機是一常見的做法,用于控制復雜的數字系統行為。狀態機能夠根據當前的輸入和系統狀態,決定下一步的動作和新的
    的頭像 發表于 07-18 15:57 ?654次閱讀
    主站蜘蛛池模板: 狼色在线视频| 日本不卡一区视频| 黄色一区二区三区| 成人黄色一级片| vvvv98国产成人综合青青| 亚洲一区二区在线播放| 久久久久国产成人精品亚洲午夜| 综合aⅴ| 亚洲成年人在线| 日韩精品毛片| 激情五月综合婷婷| 最色网在线观看| 免费看黄色的网站| 欧美黑人性xxx猛交| 黄免费网站| 色妞在线| 国产一级大片在线观看| 午夜骚片| 澳门久久| 国产三级观看久久| 色激情网| 久久国产高清视频| 68日本xxxⅹxxxxx18| 女性私密部位扒开的视频 | 国产91久久最新观看地址| 免费一看一级毛片| 高清性色生活片欧美在线| 天天草天天操| 欧美ol丝袜高跟秘书在线观看| 99热1| 欧美一级免费在线观看| 波多野结衣在线一区| 濑亚美莉vs黑人欧美视频| 色播亚洲| 91大神视频网站| 性香港xxxxx免费视频播放| 你懂得在线| 午夜免费视频| 欧美色淫网站免费观看| 亚洲国产成人久久| 久草色播|