C 語言函數(shù)的調(diào)用
在一個(gè)程序的編寫過程中,隨著代碼量的增加,如果把所有的語句都寫到 main 函數(shù)中,一方面程序會(huì)顯得的比較亂,另外一個(gè)方面,當(dāng)同一個(gè)功能需要在不同地方執(zhí)行時(shí),我們就得再重復(fù)寫一遍相同的語句。此時(shí),如果把一些零碎的功能單獨(dú)寫成一個(gè)函數(shù),在需要它們時(shí)只需進(jìn)行一些簡單的函數(shù)調(diào)用,這樣既有助于程序結(jié)構(gòu)的清晰條理,又可以避免大塊的代碼重復(fù)。
在實(shí)際工程項(xiàng)目中,一個(gè)程序通常都是由很多個(gè)子程序模塊組成的,一個(gè)模塊實(shí)現(xiàn)一個(gè)特定的功能,在 C 語言中,這個(gè)模塊就用函數(shù)來表示。一個(gè) C 程序一般由一個(gè)主函數(shù)和若干個(gè)其他函數(shù)構(gòu)成。主函數(shù)可以調(diào)用其它函數(shù),其它函數(shù)也可以相互調(diào)用,但其它函數(shù)不能調(diào)用主函數(shù)。在我們的51單片機(jī)程序中,還有中斷服務(wù)函數(shù),是當(dāng)相應(yīng)的中斷到來后自動(dòng)調(diào)用的,不需要也不能由其它函數(shù)來調(diào)用。
函數(shù)調(diào)用的一般形式是:
函數(shù)名(實(shí)參列表);
函數(shù)名就是需要調(diào)用的函數(shù)的名稱,實(shí)參列表就是根據(jù)實(shí)際需求調(diào)用函數(shù)要傳遞給被調(diào)用函數(shù)的參數(shù)列表,不需要傳遞參數(shù)時(shí)只保留括號就可以了,傳遞多個(gè)參數(shù)時(shí)參數(shù)之間要用逗號隔開。
那么我先舉例看一下函數(shù)調(diào)用使程序結(jié)構(gòu)更加條理清晰方面的作用。回顧一下圖6-1所示的程序流程圖和為實(shí)現(xiàn)它而編寫的程序代碼,相對來說這個(gè)主函數(shù)的結(jié)構(gòu)就比較復(fù)雜了,
很難一眼看清楚它的執(zhí)行流程。那么如果我們把其中最重要的兩件事——秒計(jì)數(shù)和數(shù)碼管動(dòng)態(tài)掃描功能都用單獨(dú)的函數(shù)來實(shí)現(xiàn)會(huì)怎樣呢?來看程序。
#includesbitADDR0=P1^0;sbitADDR1=P1^1;sbitADDR2=P1^2;sbitADDR3=P1^3;sbitENLED=P1^4;unsignedcharcodeLedChar[]={//數(shù)碼管顯示字符轉(zhuǎn)換表0xC0,0xF9,0xA4,0xB0,0x99,0x92,0x82,0xF8,0x80,0x90,0x88,0x83,0xC6,0xA1,0x86,0x8E};unsignedcharLedBuff[6]={//數(shù)碼管顯示緩沖區(qū),初值0xFF確保啟動(dòng)時(shí)都不亮0xFF,0xFF,0xFF,0xFF,0xFF,0xFF};voidSecondCount();voidLedRefresh();voidmain(){ENLED=0;//使能U3,選擇控制數(shù)碼管ADDR3=1;//因?yàn)樾枰獎(jiǎng)討B(tài)改變ADDR0-2的值,所以不需要再初始化了TMOD=0x01;//設(shè)置T0為模式1TH0=0xFC;//為T0賦初值0xFC67,定時(shí)1msTL0=0x67;TR0=1;//啟動(dòng)T0while(1){if(TF0==1){//判斷T0是否溢出TF0=0;//T0溢出后,清零中斷標(biāo)志TH0=0xFC;//并重新賦初值TL0=0x67;SecondCount();//調(diào)用秒計(jì)數(shù)函數(shù)LedRefresh();//調(diào)用顯示刷新函數(shù)}}}/*秒計(jì)數(shù)函數(shù),每秒進(jìn)行一次秒數(shù)+1,并轉(zhuǎn)換為數(shù)碼管顯示字符*/voidSecondCount(){staticunsignedintcnt=0;//記錄T0中斷次數(shù)staticunsignedlongsec=0;//記錄經(jīng)過的秒數(shù)cnt++;//計(jì)數(shù)值自加1if(cnt>=1000){//判斷T0溢出是否達(dá)到1000次cnt=0;//達(dá)到1000次后計(jì)數(shù)值清零sec++;//秒計(jì)數(shù)自加1LedBuff[0]=LedChar[sec%10];LedBuff[1]=LedChar[sec/10%10];LedBuff[2]=LedChar[sec/100%10];LedBuff[3]=LedChar[sec/1000%10];LedBuff[4]=LedChar[sec/10000%10];LedBuff[5]=LedChar[sec/100000%10];}}/*數(shù)碼管動(dòng)態(tài)掃描刷新函數(shù)*/voidLedRefresh(){staticunsignedchari=0;//動(dòng)態(tài)掃描的索引switch(i){case0:ADDR2=0;ADDR1=0;ADDR0=0;i++;P0=LedBuff[0];break;case1:ADDR2=0;ADDR1=0;ADDR0=1;i++;P0=LedBuff[1];break;case2:ADDR2=0;ADDR1=1;ADDR0=0;i++;P0=LedBuff[2];break;case3:ADDR2=0;ADDR1=1;ADDR0=1;i++;P0=LedBuff[3];break;case4:ADDR2=1;ADDR1=0;ADDR0=0;i++;P0=LedBuff[4];break;case5:ADDR2=1;ADDR1=0;ADDR0=1;i=0;P0=LedBuff[5];break;default:break;}}
看一下,主函數(shù)的結(jié)構(gòu)是不是清晰的多了——每隔 1 ms 就去干兩件事,至于這兩件事是什么交由各自的函數(shù)去實(shí)現(xiàn)。還請大家注意一點(diǎn):原來程序中的 i、cnt、sec 這三個(gè)變量在放到單獨(dú)的函數(shù)中后,都加了 static 關(guān)鍵字而變成了靜態(tài)變量。因?yàn)樵瓉淼?main()永遠(yuǎn)不會(huì)結(jié)束所以它們的值也總是得到保持的,但現(xiàn)在它們在各自的功能函數(shù)內(nèi),如不加 static 修飾那么每次函數(shù)被調(diào)用時(shí)它們的值就都成了初值了,借此也把靜態(tài)變量再加深一下理解吧。
當(dāng)然,這是我們刻意把程序功能做了這樣的劃分,主要目的還是來講解函數(shù)的調(diào)用,對于這個(gè)程序即使你不劃分函數(shù)也復(fù)雜不到哪里去,但繼續(xù)學(xué)下去你就能領(lǐng)會(huì)到劃分功能函數(shù)的必要了。現(xiàn)在我們還是把注意力放在學(xué)習(xí)函數(shù)調(diào)用上,有以下幾點(diǎn)需要大家注意:
1) 函數(shù)調(diào)用的時(shí)候,不需要加函數(shù)類型。我們在主函數(shù)內(nèi)調(diào)用 SecondCount()和 LedRefresh()時(shí)都沒有加 void。
2) 調(diào)用函數(shù)與被調(diào)用函數(shù)的位置關(guān)系,C 語言規(guī)定:函數(shù)在被調(diào)用之前,必須先被定義或聲明。意思就是說:在一個(gè)文件中,一個(gè)函數(shù)應(yīng)該先定義,然后才能被調(diào)用,也就是調(diào)用函數(shù)應(yīng)位于被調(diào)用函數(shù)的下方。但是作為一種通常的編程規(guī)范,我們推薦 main 函數(shù)寫在最前面(因?yàn)樗鸬教峋V挈領(lǐng)的作用),其后再定義各個(gè)功能函數(shù),而中斷函數(shù)則寫在文件的最后。那么主函數(shù)要調(diào)用定義在它之后的函數(shù)怎么辦呢?我們就在文件開頭,所有函數(shù)定義之前,開辟一塊區(qū)域,叫做函數(shù)聲明區(qū),用來把被調(diào)用的函數(shù)聲明一下,如此,該函數(shù)就可以被隨意調(diào)用了。如上述例程所示。
3) 函數(shù)聲明的時(shí)候必須加函數(shù)類型,函數(shù)的形式參數(shù),最后加上一個(gè)分號表示結(jié)束。函數(shù)聲明行與函數(shù)定義行的唯一區(qū)別就是最后的分號,其它的都必須保持一致。這點(diǎn)請尤其注意,初學(xué)者很容易因粗心大意而搞錯(cuò)分號或是修改了定義行中的形參卻忘了修改聲明行中的形參,導(dǎo)致程序編譯不過。