在C語言編程過程中,由于計(jì)算需要,會(huì)使用各種各樣的變量,用于給需要訪問的地址取個(gè)名稱,方便編程中使用,代碼維護(hù)者也容易理解。
這里先給大家分享一個(gè)案例,讓大家意識(shí)到變量賦初始值的重要性。
某用戶在基于瑞薩的MCU:RA6T2做開發(fā)時(shí),發(fā)現(xiàn)一個(gè)問題,MCU發(fā)出的CAN數(shù)據(jù)幀總是莫名其妙的出錯(cuò),比如應(yīng)用中明明只使用了CAN的擴(kuò)展幀,但是使用捕捉工具總是能捕捉到遠(yuǎn)程幀,出現(xiàn)遠(yuǎn)程幀的情況毫無規(guī)律可言,有時(shí)添加一個(gè)定時(shí)器中斷,該現(xiàn)象就不會(huì)出現(xiàn)了,有時(shí)修改了代碼里某處跟CAN沒有任何關(guān)系的代碼,該問題又會(huì)出現(xiàn),過了兩周時(shí)間調(diào)試無果。在介入Debug時(shí)發(fā)現(xiàn),他使用的是CAN擴(kuò)展幀,擴(kuò)展幀使用29位ID標(biāo)識(shí)符,而且對(duì)ID區(qū)數(shù)據(jù)定義了一個(gè)如下結(jié)構(gòu)體:
他在需要發(fā)送CAN幀時(shí),申請(qǐng)一個(gè)如上結(jié)構(gòu)體的臨時(shí)變量can_id,在把can_id.id賦值后,再把該變量的地址傳遞給CAN的發(fā)送函數(shù),在發(fā)送函數(shù)里使用如下語句把id的數(shù)據(jù)寫入CAN的發(fā)送消息緩沖寄存器:
如下圖,其中第30位的0表示數(shù)據(jù)幀,并不是遠(yuǎn)程幀,31位的1表示擴(kuò)展幀。
用戶是把can_id的所有數(shù)據(jù)賦值給了CFDTMID0寄存器,假如can_id.dummy中第二個(gè)位是1,會(huì)有什么后果呢?CFDTMID0.TMRTR=1,即CAN會(huì)發(fā)送遠(yuǎn)程幀。
用戶又問:我沒有給dummy賦值啊,為什么dummy的第二個(gè)位會(huì)變成1呢?這就是問題所在了,就是因?yàn)樗麤]有給can_id.dummy賦值,所以can_id.dummy有可能為任意的值。下面詳細(xì)分析一下,為什么這個(gè)局部變量的值會(huì)隨意變化。
大家知道,變量根據(jù)存儲(chǔ)類型和用途,一般可以分成:全局變量和局部變量。全局變量,就是指分配了固定地址的變量,全局變量可以在整個(gè)代碼范圍內(nèi)使用。我們?cè)谏暾?qǐng)全局變量時(shí),有時(shí)對(duì)它賦一個(gè)初始值,也時(shí)也不會(huì)賦初始值,在代碼上可能看不出有什么區(qū)別,但是編譯器在編譯程序時(shí),是區(qū)別對(duì)待他們的。對(duì)于有初始化的變量,編譯器還需要在Code Flash里(代碼存儲(chǔ)區(qū))分配一段空間,把變量的初始值全部存儲(chǔ)在該區(qū)域里,并且在MCU的啟動(dòng)代碼里插入一段程序,把這些Code Flash區(qū)的初始值拷貝到變量對(duì)應(yīng)的RAM地址中。假如上面的can_id是全局變量,并且申明變量的同時(shí)并按下圖賦初始值:
這時(shí)can_id.dummy=0,如果代碼中用戶沒有再賦值,它的值也不會(huì)變化,這樣就不會(huì)發(fā)生用戶的那個(gè)遠(yuǎn)程幀的問題了。對(duì)于沒有賦初始值的全局變量,編譯器只是分配RAM的地址,并不會(huì)修改RAM地址里的數(shù)據(jù),那么這個(gè)變量的值就會(huì)依賴于MCU啟動(dòng)時(shí)RAM里的值了。為了避免未賦值的全局變量出現(xiàn)上述的問題,我們一般會(huì)在MCU啟動(dòng)代碼里插入未賦初始值全局變量的清零操作,相當(dāng)于做了一個(gè)未賦初始值的全局變量的初始化賦值操作。
像上面的案例,can_id申請(qǐng)的是局部變量,這又是什么情況呢?
因?yàn)镸CU的RAM資源有限,為了最大限度的利用RAM,MCU會(huì)提前分配一塊RAM區(qū)域,叫堆棧區(qū),這塊區(qū)域大家共用,對(duì)于只需要在某個(gè)函數(shù)內(nèi)使用的變量,引入了局部變量概念。在開始執(zhí)行該函數(shù)時(shí),才從堆棧里分配地址給局部變量使用,函數(shù)執(zhí)行結(jié)束后,該變量占用的RAM區(qū)域被堆?;厥眨?dāng)下次再調(diào)用該函數(shù),再重新分配RAM。因此對(duì)于局部變量,每次申請(qǐng)到的地址是不同的,該地址很可能是其它函數(shù)使用過并改寫數(shù)據(jù)了的,因此每次函數(shù)調(diào)用時(shí)can_id.dummy的數(shù)據(jù)是不確定的。因?yàn)槎褩^(qū)里的數(shù)據(jù)是被反復(fù)利用的,即使MCU的初始化代碼對(duì)堆棧區(qū)域做清零處理,也是沒有意義的。
由此看來,局部變量在申請(qǐng)的時(shí)候賦一個(gè)初始值,是非常有必要的。雖然有時(shí)候賦初始值沒有用,但是出現(xiàn)問題時(shí)常常是致命的,而且也是非常難以定位的,你可能覺得我的代碼里后面肯定會(huì)賦值的,但是后面維護(hù)該項(xiàng)目的其他工程師并不一定意識(shí)到這一點(diǎn)。像類似上面的案例,我在其他用戶當(dāng)中也是經(jīng)常見到的。因此軟件工程師在編程的時(shí)候,一定要養(yǎng)成局部變量賦初始值的習(xí)慣。
來源:瑞薩MCU小百科
免責(zé)聲明:本文為轉(zhuǎn)載文章,轉(zhuǎn)載此文目的在于傳遞更多信息,版權(quán)歸原作者所有。本文所用視頻、圖片、文字如涉及作品版權(quán)問題,請(qǐng)聯(lián)系小編進(jìn)行處理
審核編輯 黃宇
-
mcu
+關(guān)注
關(guān)注
146文章
17981瀏覽量
366792 -
CAN
+關(guān)注
關(guān)注
57文章
2920瀏覽量
467785 -
編程
+關(guān)注
關(guān)注
88文章
3689瀏覽量
95245 -
變量
+關(guān)注
關(guān)注
0文章
614瀏覽量
28951
發(fā)布評(píng)論請(qǐng)先 登錄
連接器氣密性檢測(cè)的重要性

ADS1298 CONFIG2讀取初始值不正常是什么原因引起的?
構(gòu)建綜合指揮調(diào)度系統(tǒng)的重要性
TLC5615芯片輸出的初始值是不是為0?
EE-88:使用21xx編譯器在C中初始化變量

電橋在電子測(cè)試中的重要性
PCB板元器件點(diǎn)膠加固的重要性

PROM器件在物聯(lián)網(wǎng)設(shè)備中的重要性
?選對(duì)波長(zhǎng)對(duì)在固化時(shí)候的重要性

元器件在電路設(shè)計(jì)中的重要性
plc編程st語言怎么編
關(guān)于蓄電池氣密性檢測(cè)儀重要性和使用方法

評(píng)論