單片機(jī)程序結(jié)構(gòu)再分析
在學(xué)C++時(shí)對(duì)對(duì)單片機(jī)程序有一些新的想法。
在《單片機(jī)用定時(shí)器分配任務(wù)程序結(jié)構(gòu)總結(jié)》里面,把整個(gè)系統(tǒng)分為兩個(gè)進(jìn)程:主函數(shù)和主函數(shù)調(diào)用的所有函數(shù),這是主進(jìn)程;還有中斷觸發(fā)的一個(gè)進(jìn)程。
各種中斷的到來(lái)會(huì)立刻讓主進(jìn)程相關(guān)數(shù)據(jù)入棧保存,然后開(kāi)始一段新的代碼,執(zhí)行完成后再?gòu)亩褩V凶x取數(shù)據(jù)返回原來(lái)的地方繼續(xù)執(zhí)行,這種切換方式其實(shí)就和操作系統(tǒng)的各個(gè)進(jìn)程間切換是一模一樣的。所以把它們說(shuō)成是兩個(gè)進(jìn)程確實(shí)非常貼切。
現(xiàn)在,在主進(jìn)程中進(jìn)一步把函數(shù)分為兩類:實(shí)現(xiàn)算法和邏輯功能的函數(shù),以及公共函數(shù)。
先看下面這幅圖吧(取自譚浩強(qiáng) C++程序設(shè)計(jì)P227)
這里面所有函數(shù)都是由主函數(shù)調(diào)用的,屬于主進(jìn)程,并且列出來(lái)的所有函數(shù)都體現(xiàn)了算法,也就是用于構(gòu)成邏輯結(jié)構(gòu)。
例如在函數(shù)1里面想進(jìn)入函數(shù)2,不是直接調(diào)用函數(shù)2,而是先返回函數(shù)1,再由主循環(huán)分配到函數(shù)2。
這種程序結(jié)構(gòu)特別適合于多種“界面”的功能,比如電子鐘里面的時(shí)鐘顯示界面和設(shè)置界面,就是兩個(gè)函數(shù),進(jìn)去了之后就執(zhí)行這個(gè)函數(shù)的特定的功能。再比如DYS388的顯示方式,有16位全彩顯示和7色顯示兩種模式,這兩種顯示模式就是兩個(gè)函數(shù),進(jìn)入某一種顯示模式后就會(huì)以那種顯示模式特定的顯示方式進(jìn)行顯示。一般情況下,主進(jìn)程不會(huì)停留在主循環(huán)里,而是偶爾退出到主循環(huán)重新分配下一個(gè)將要進(jìn)入的函數(shù)。
這些函數(shù)之間有一些公共變量,也有一些于函數(shù)對(duì)應(yīng)的用于完成特定功能的變量。比如 DYS388中16位刷新函數(shù)和7色刷新函數(shù)都對(duì)應(yīng)一段自己的顯存,這些顯存是有特定用處的,一般其它函數(shù)不會(huì)使用(但確實(shí)是公共變量,是可以被使用的);也有一些變量作用就是被各個(gè)函數(shù)使用,甚至用于函數(shù)間通信,輔助完成這些函數(shù)之間的邏輯結(jié)構(gòu)的構(gòu)建,比如DYS388中的界面標(biāo)志變量DispMode,這個(gè)標(biāo)志變量就指明了當(dāng)前工作于那種刷新方式,任何函數(shù)(包括中斷進(jìn)程中的函數(shù))都可以通過(guò)改變此變量來(lái)切換顯示模式。
而今天我要說(shuō)的不只是這些,上面說(shuō)的是變量,有些變量對(duì)應(yīng)特定的函數(shù)使用,有些變量可以被所有函數(shù)使用。
與之對(duì)應(yīng)的還有函數(shù),圖中畫出的函數(shù)都是所謂的“界面函數(shù)”(自己起的名字哈),用于完成某一特定任務(wù)的函數(shù),一般進(jìn)入這個(gè)函數(shù)后主進(jìn)程就會(huì)停在里面,當(dāng)達(dá)到特殊目的后返回。而這些“界面函數(shù)”也會(huì)不斷地調(diào)用其它函數(shù)完成功能,比如延時(shí)等。
這些被界面函數(shù)調(diào)用的函數(shù)把它們稱作“工具函數(shù)”。這些功能函數(shù)中有一些是公用的,比如延時(shí)函數(shù),很多地方都會(huì)用到。而也有一些是某一個(gè)界面函數(shù)才會(huì)用到的,用于完成這個(gè)特殊功能的函數(shù),比如DYS388中的一行的掃描程序,16位顯示函數(shù)不斷調(diào)用行掃描函數(shù)從而完成整屏的刷新。
這樣,這些所謂的“工具函數(shù)”就和變量對(duì)應(yīng)起來(lái)了。整體的程序框架是由各個(gè)“界面函數(shù)”和少數(shù)關(guān)鍵的全局變量構(gòu)建起來(lái)的。為這個(gè)框架服務(wù)的還有其它一些變量和工具函數(shù),有些變量為特定的界面函數(shù)服務(wù),有些則可為所有函數(shù)使用;有些工具函數(shù)為特定的界面函數(shù)調(diào)用,有些工具函數(shù)則可被所有的界面函數(shù)調(diào)用。
到此還沒(méi)有結(jié)束,上面只考慮了主進(jìn)程,而中斷也會(huì)開(kāi)辟一條進(jìn)程,這個(gè)進(jìn)程中也可能會(huì)有類似主進(jìn)程的結(jié)構(gòu),雖然在實(shí)際使用中單片機(jī)中斷程序一般比較簡(jiǎn)單,不會(huì)有太復(fù)雜的結(jié)構(gòu),因?yàn)橹袛嗵幚沓绦蛲顺龊?,里面的局部變量不?huì)想主進(jìn)程那樣被保存下來(lái),中斷處理程序只能靠全局變量進(jìn)行記憶。However,中斷處理程序毫無(wú)疑問(wèn)地可以使用上面定義的所有全局變量和函數(shù)。
在這里我想說(shuō)的是,當(dāng)一個(gè)進(jìn)程調(diào)用另一個(gè)進(jìn)程會(huì)使用的函數(shù)(函數(shù)A)時(shí)一定要小心,因?yàn)檫@個(gè)進(jìn)程是由中斷開(kāi)辟的(至少在單片機(jī)里面是),而這個(gè)中斷可能正是從將要調(diào)用的函數(shù)A中跳出來(lái)的,即使不是從即將調(diào)用的函數(shù)A中跳出(假設(shè)從函數(shù)B中跳出),也可能函數(shù)A會(huì)調(diào)用函數(shù)B。
這些都會(huì)導(dǎo)致單片機(jī)死機(jī)的,編譯時(shí)也應(yīng)該會(huì)有警告的。
總結(jié)一下,這篇文章主要想說(shuō)如下內(nèi)容:
整個(gè)主進(jìn)程的框架是由“界面函數(shù)”和一些關(guān)鍵的全局變量構(gòu)成的。有其它的變量和函數(shù)為它們服務(wù),有些變量和函數(shù)是為了輔助某一個(gè)界面函數(shù)完成特殊功能,其它函數(shù)一般不會(huì)用到;也有些變量和函數(shù)位全局服務(wù)的,完成一些通用的功能。
除主進(jìn)程外,由中斷開(kāi)辟的另一道進(jìn)程也可能會(huì)有為自己服務(wù)的變量和函數(shù),當(dāng)然也可以調(diào)用主進(jìn)程中的變量和函數(shù),利用他們?yōu)樽约悍?wù),或者用于跟主進(jìn)程通信。而在中斷進(jìn)程調(diào)用主進(jìn)程的函數(shù)時(shí)一定要注意一個(gè)原則:不要讓調(diào)用的函數(shù)調(diào)用到被中斷的函數(shù)。必要時(shí)可以為中斷進(jìn)程單獨(dú)寫一個(gè)服務(wù)函數(shù),函數(shù)內(nèi)容可能跟主進(jìn)程中的某個(gè)函數(shù)一模一樣,但這樣可以避免上述問(wèn)題。
編輯:admin 最后修改時(shí)間:2018-05-18