《電子技術應用》
您所在的位置:首頁 > 可編程邏輯 > 業界動態 > 教學:單片機狀態機編程詳解

教學:單片機狀態機編程詳解

2022-08-31
來源:單片機與嵌入式
關鍵詞: 單片機 狀態機

  玩單片機還可以,各個外設也都會驅動,但是如果讓你完整的寫一套代碼時,卻無邏輯與框架可言。這說明編程還處于比較低的水平,你需要學會一種好的編程框架或者一種編程思想!比如模塊化編程、狀態機編程、分層思想等,相關推薦:分享兩種單片機編程思想。

  本文來說一下狀態機編程。

  什么是狀態機?

  狀態機(state machine)有5個要素:

  狀態(state)

  遷移(transiTIon)

  事件(event)

  動作(acTIon)

  條件(guard)

  狀態:一個系統在某一時刻所存在的穩定的工作情況,系統在整個工作周期中可能有多個狀態。例如一部電動機共有正轉、反轉、停轉這 3 種狀態。

  一個狀態機需要在狀態集合中選取一個狀態作為初始狀態。

  遷移:系統從一個狀態轉移到另一個狀態的過程稱作遷移,遷移不是自動發生的,需要外界對系統施加影響。停轉的電動機自己不會轉起來,讓它轉起來必須上電。

  事件:某一時刻發生的對系統有意義的事情,狀態機之所以發生狀態遷移,就是因為出現了事件。對電動機來講,加正電壓、加負電壓、斷電就是事件。

  動作:在狀態機的遷移過程中,狀態機會做出一些其它的行為,這些行為就是動作,動作是狀態機對事件的響應。給停轉的電動機加正電壓,電動機由停轉狀態遷移到正轉狀態,同時會啟動電機,這個啟動過程可以看做是動作,也就是對上電事件的響應。

  條件:狀態機對事件并不是有求必應的,有了事件,狀態機還要滿足一定的條件才能發生狀態遷移。還是以停轉狀態的電動機為例,雖然合閘上電了,但是如果供電線路有問題的話,電動機還是不能轉起來。

  舉個例子

  要解決的問題

  電路如下圖:

  器件包括單片機MCU、一按鍵K0、LED燈L1和L2。

  實現功能描述:

  L1L2狀態轉換順序OFF/OFF--->ON/OFF--->ON/ON--->OFF/ON--->OFF/OFF

  通過按鍵控制L1L2的狀態,每次狀態轉換需連續按鍵5次

  L1L2的初始狀態OFF/OFF

 

3a579106-26f8-11ed-ba43-dac502259ad0.png

  狀態轉換圖

  在狀態機編程中,正確的順序應該是先有狀態轉換圖,后有程序,程序應該是根據設計好的狀態圖寫出來的。

  下面這張按鍵控制流水燈狀態轉換圖,是用UML(統一建模語言)的語法元素畫出來的,語法不是很標準,但拿來解釋問題足夠了。

3a677ad0-26f8-11ed-ba43-dac502259ad0.png

  上圖中,圓角矩形代表狀態機的各個狀態,里面標注著狀態的名稱。

  帶箭頭的直線或弧線代表狀態遷移,起于初態,止于次態。

  圖中的文字內容是對遷移的說明,格式是:事件[條件]/動作列表(后兩項可選)。

  “事件[條件]/動作列表”要說明的意思是:如果在某個狀態下發生了“事件”,并且狀態機

  滿足“[條件]”,那么就要執行此次狀態轉移,同時要產生一系列“動作”,以響應事件。在這個例子里,我用“KEY”表示擊鍵事件。

  圖中有一個黑色實心圓點,表示狀態機在工作之前所處的一種不可知的狀態,在運行之前狀態機必須強制地由這個狀態遷移到初始狀態,這個遷移可以有動作列表(如圖1所示),但不需要事件觸發。

  圖中還有一個包含黑色實心圓點的圓圈,表示狀態機生命周期的結束,這個例子中的狀態機生生不息,所以沒有狀態指向該圓圈。

  程序代碼

  下面是根據上述狀態轉換圖寫成的代碼:

  void main(void){ sys_init(); led_off(LED1); led_off(LED2); g_stFSM.u8LedStat = LS_OFFOFF; g_stFSM.u8KeyCnt = 0;while(1) {if(test_key()==TRUE)  {   fsm_acTIve();  }else  {   ; /*idle code*/  } }}void fsm_acTIve(void){if(g_stFSM.u8KeyCnt > 3) /*擊鍵是否滿 5 次*/ {switch(g_stFSM.u8LedStat)  {case LS_OFFOFF:    led_on(LED1); /*輸出動作*/    g_stFSM.u8KeyCnt = 0;    g_stFSM.u8LedStat = LS_ONOFF; /*狀態遷移*/break;case LS_ONOFF:    led_on(LED2); /*輸出動作*/    g_stFSM.u8KeyCnt = 0;    g_stFSM.u8LedStat = LS_ONON; /*狀態遷移*/break;case LS_ONON:    led_off(LED1); /*輸出動作*/    g_stFSM.u8KeyCnt = 0;    g_stFSM.u8LedStat = LS_OFFON; /*狀態遷移*/break;case LS_OFFON:    led_off(LED2); /*輸出動作*/    g_stFSM.u8KeyCnt = 0;    g_stFSM.u8LedStat = LS_OFFOFF; /*狀態遷移*/break;default: /*非法狀態*/    led_off(LED1);    led_off(LED2);    g_stFSM.u8KeyCnt = 0;    g_stFSM.u8LedStat = LS_OFFOFF; /*恢復初始狀態*/break;  } }else {  g_stFSM.u8KeyCnt++; /*狀態不遷移,僅記錄擊鍵次數*/ }}

  先看一下fsm_active()這個函數,g_stFSM.u8KeyCnt = 0;這個語句在switch—case里共出現了 5 次,前 4 次是作為各個狀態遷移的動作出現的。從代碼簡化提高效率的角度來看,我們完全可以把這 5 次合并為 1 次放在 switch—case 語句之前,兩者的效果是完全一樣的,代碼里之所以這樣啰嗦,是為了清晰地表明每次狀態遷移中所有的動作細節,這種方式和上面狀態轉換圖所要表達的意圖是完全一致的。

  再看一下g_stFSM這個狀態機結構體變量,它有兩個成員:u8LedStat和 u8KeyCnt。用這個結構體來做狀態機好像有點兒啰嗦,我們能不能只用一個像 u8LedStat 這樣的整型變量來做狀態機呢?

  當然可以!我們把上圖中的這 4 個狀態各自拆分成 5 個小狀態,這樣用 20 個狀態同樣能實現這個狀態機,而且只需要一個 unsigned char 型的變量就足夠了,每次擊鍵都會引發狀態遷移, 每遷移 5 次就能改變一次 LED 燈的狀態,從外面看兩種方法的效果完全一樣。

  假設我把功能要求改一下,把連續擊鍵5次改變L1L2的狀態改為連續擊鍵100次才能改變L1L2的狀態。這樣的話第二種方法需要4X100=400個狀態!而且函數fsm_active()中的switch—case語句里要有400個case,這樣的程序還有法兒寫么?!

  同樣的功能改動,如果用g_stFSM這個結構體來實現狀態機的話,函數fsm_active()只需要將if(g_stFSM.u8KeyCnt>3)改為if(g_stFSM.u8KeyCnt > 98)就可以了!

  g_stFSM結構體的兩個成員中,u8LedStat可以看作是質變因子,相當于主變量;u8KeyCnt可以看作是量變因子,相當于輔助變量。量變因子的逐步積累會引發質變因子的變化。

  像g_stFSM這樣的狀態機被稱作Extended State Machine。



更多信息可以來這里獲取==>>電子技術應用-AET<< 

mmexport1621241704608.jpg

本站內容除特別聲明的原創文章之外,轉載內容只為傳遞更多信息,并不代表本網站贊同其觀點。轉載的所有的文章、圖片、音/視頻文件等資料的版權歸版權所有權人所有。本站采用的非本站原創文章及圖片等內容無法一一聯系確認版權者。如涉及作品內容、版權和其它問題,請及時通過電子郵件或電話通知我們,以便迅速采取適當措施,避免給雙方造成不必要的經濟損失。聯系電話:010-82306118;郵箱:aet@chinaaet.com。
亚洲一区二区欧美_亚洲丝袜一区_99re亚洲国产精品_日韩亚洲一区二区
欧美诱惑福利视频| 老司机久久99久久精品播放免费| 亚洲一区二区三区视频播放| 亚洲日本成人在线观看| 在线观看成人小视频| 国产日产欧美a一级在线| 欧美午夜性色大片在线观看| 欧美噜噜久久久xxx| 欧美大片一区二区| 亚洲午夜影视影院在线观看| 欧美电影在线观看| 亚洲一区二区三区777| 亚洲精品字幕| 亚洲人人精品| 亚洲啪啪91| 亚洲精品在线三区| 亚洲欧洲一区二区在线播放| 欧美日本一区| 国产综合精品| 亚洲综合久久久久| 日韩视频精品在线观看| 亚洲国产老妈| 在线观看一区二区视频| 亚洲电影欧美电影有声小说| 在线日韩成人| 在线色欧美三级视频| 久久精品国产清自在天天线| 国产一区视频在线看| 国产一级揄自揄精品视频| 国产三级欧美三级日产三级99| 国产精品视频yy9299一区| 欧美午夜理伦三级在线观看| 国产精品国产自产拍高清av王其| 国产精品海角社区在线观看| 国产精品久久久久一区| 国产麻豆视频精品| 好吊妞**欧美| 亚洲黄色免费电影| 制服诱惑一区二区| 性欧美暴力猛交69hd| 久久精彩免费视频| 日韩一区二区免费高清| 亚洲天堂久久| 欧美在线www| 美女精品国产| 欧美日韩三级在线| 国产精品综合网站| 樱桃成人精品视频在线播放| 亚洲精品久久久久中文字幕欢迎你| 一区二区久久久久久| 亚洲欧美国产毛片在线| 亚洲国产乱码最新视频| 一区二区三区日韩精品视频| 欧美一激情一区二区三区| 久久色在线观看| 欧美精品一区二区三区四区| 国产精品久久影院| 精品成人久久| 国产精品99久久久久久白浆小说| 性欧美1819sex性高清| 亚洲精品孕妇| 欧美一区成人| 欧美福利影院| 国产啪精品视频| 亚洲激情综合| 欧美一区二区三区四区夜夜大片 | 亚洲国产高清高潮精品美女| 亚洲最新视频在线| 久久精品视频免费| 亚洲一区二区三区四区视频 | 国产精品一区二区三区观看 | 亚洲女同精品视频| 亚洲欧洲一区二区天堂久久| 亚洲专区在线视频| 农村妇女精品| 国产精品爽爽ⅴa在线观看| 亚洲第一黄网| 午夜精品久久久久久99热软件| 亚洲区免费影片| 欧美在线1区| 欧美天堂在线观看| 亚洲国产精品999| 性欧美1819性猛交| 亚洲午夜久久久久久尤物 | 伊甸园精品99久久久久久| 野花国产精品入口| 亚洲激情欧美激情| 久久高清免费观看| 欧美新色视频| 亚洲人精品午夜在线观看| 亚洲成人在线网站| 久久国产夜色精品鲁鲁99| 欧美日韩在线电影| 91久久黄色| 亚洲成色777777在线观看影院| 亚洲欧美中文另类| 欧美日在线观看| 亚洲精品乱码久久久久久久久| 久久激五月天综合精品| 欧美一二三区精品| 国产精品久久久久久久电影| 亚洲蜜桃精久久久久久久| 亚洲精品一区二区三区樱花| 老司机精品视频网站| 国产一区二区三区黄视频| 亚洲一区bb| 亚洲一区二区三区三| 欧美日本乱大交xxxxx| 亚洲第一福利视频| 亚洲激情偷拍| 美女免费视频一区| 狠狠色丁香婷婷综合| 欧美一区二区三区婷婷月色| 欧美一区二区视频免费观看| 国产精品女主播一区二区三区| 夜夜嗨av一区二区三区中文字幕| 亚洲美女色禁图| 欧美国产亚洲精品久久久8v| 在线观看免费视频综合| 亚洲福利专区| 久久婷婷色综合| 韩国三级电影久久久久久| 性色一区二区三区| 久久成人免费日本黄色| 国产欧美日韩亚洲精品| 先锋影院在线亚洲| 久久久www| 在线观看国产日韩| 日韩视频在线一区二区| 久久国内精品自在自线400部| 国产精品久久久久77777| 99re亚洲国产精品| 亚洲一区二区三区中文字幕| 欧美性猛片xxxx免费看久爱| 在线午夜精品自拍| 午夜精品偷拍| 国产欧美韩国高清| 久久xxxx| 美女精品视频一区| 老司机免费视频一区二区三区 | 欧美日韩亚洲另类| 这里只有精品在线播放| 亚洲欧美在线看| 国产视频一区二区三区在线观看| 性色av一区二区三区| 美女黄网久久| 亚洲精品国产精品乱码不99按摩| 99re这里只有精品6| 欧美日韩一区在线观看视频| 亚洲天堂免费观看| 久久久久久久久久久一区 | 亚洲欧洲在线一区| 这里只有精品视频在线| 国产精品日韩欧美一区二区| 欧美一区二区精美| 欧美jizzhd精品欧美巨大免费| 亚洲精品国偷自产在线99热| 亚洲在线免费观看| 国产一区二区激情| 亚洲精品一区二区三区蜜桃久| 欧美日韩一区二区三区四区在线观看| 亚洲女同性videos| 欧美91精品| 亚洲视频图片小说| 卡一卡二国产精品| 99精品欧美一区二区三区综合在线| 亚洲欧美综合网| 一区免费视频| 亚洲午夜影视影院在线观看| 国产一区导航| 99国产精品久久久久久久| 国产精品免费久久久久久| 亚洲大胆人体在线| 欧美日韩性视频在线| 欧美亚洲在线观看| 欧美人与禽猛交乱配视频| 午夜精品成人在线视频| 欧美精品成人一区二区在线观看| 亚洲一区免费网站| 欧美高清不卡在线| 欧美一级淫片播放口| 欧美日本韩国| 久久国产黑丝| 欧美色道久久88综合亚洲精品| 欧美一级视频一区二区| 欧美一区二区三区四区视频| 91久久精品一区| 久久国产日韩欧美| 一本大道久久a久久精二百| 久久久亚洲综合| 国产精品99久久久久久人| 你懂的网址国产 欧美| 午夜天堂精品久久久久| 欧美日韩性生活视频| 亚洲国产一区二区三区青草影视| 国产精品久久久久一区二区| 亚洲精品久久久久久久久久久| 国产精品乱子久久久久| 亚洲美女视频| 伊人一区二区三区久久精品|