當(dāng)前位置:首頁 > 公眾號精選 > 21ic電子網(wǎng)
[導(dǎo)讀]許多程序員都無法正確理解C語言關(guān)鍵字volatile,這并不奇怪。因?yàn)榇蠖鄶?shù)C語言書籍通常都是一兩句一帶而過,本文將告訴你如何正確使用它。在C/C嵌入式代碼中,你是否經(jīng)歷過以下情況:代碼執(zhí)行正常–直到你打開了編譯器優(yōu)化代碼執(zhí)行正常–直到打開了中斷古怪的硬件驅(qū)動RTOS的任務(wù)獨(dú)立運(yùn)...

書本里不會講的C語言關(guān)鍵詞volatile用法


許多程序員都無法正確理解C語言關(guān)鍵字volatile,這并不奇怪。因?yàn)榇蠖鄶?shù)C語言書籍通常都是一兩句一帶而過,本文將告訴你如何正確使用它。


在C/C 嵌入式代碼中,你是否經(jīng)歷過以下情況:

  • 代碼執(zhí)行正常–直到你打開了編譯器優(yōu)化

  • 代碼執(zhí)行正常–直到打開了中斷

  • 古怪的硬件驅(qū)動

  • RTOS的任務(wù)獨(dú)立運(yùn)行正常–直到生成了其他任務(wù)


如果你的回答是“yes”,很有可能你沒有使用C語言關(guān)鍵字volatile。你并不是唯一的,很多程序員都不能正確使用volatile。不幸的是,大多數(shù)c語言書籍對volatile的藐視,只是簡單地一帶而過。


volatile用于聲明變量時的使用的限定符。它告訴編譯器該變量值可能隨時發(fā)生變化,且這種變化并不是代碼引起的。給編譯器這個暗示是很重要的。在開始前,我們向來看一看volatile的語法。


書本里不會講的C語言關(guān)鍵詞volatile用法C語言關(guān)鍵字volatile語法


聲明一個變量為volatile,可以在數(shù)據(jù)類型之前或之后加上關(guān)鍵字volatile。下面的語句,把foo聲明一個volatile的整型。


volatile int foo;

int volatile foo;


把指針指向的變量聲明為volatile很常見,尤其是I/O寄存器的地址映射。下面的語句,把pReg聲明為一個指向8-bit無符號指針,指針指向的內(nèi)容為volatile。


volatile uint8_t * pReg;

uint8_t volatile * pReg;


volatile的指針指向非volatile的變量很少見(我只使用過一次),但我還是給出相應(yīng)的語法。


int * volatile p;


順便提一下,關(guān)于為什么要在數(shù)據(jù)類型前使用volatile關(guān)鍵字,請自行百度搜素。


最后,如果你再struct或者union前使用volatile關(guān)鍵字,表明struct或者union的所有內(nèi)容都是volatile。如果這不是你的本意,可以在struct或者union成員上使用volatile關(guān)鍵字。


書本里不會講的C語言關(guān)鍵詞volatile用法正確使用C語言關(guān)鍵字volatile


只要變量可能被意外的修改,就需要把該變量聲明為volatile。在實(shí)際應(yīng)用中,只有三種類型數(shù)據(jù)可能被修改:

  • 外設(shè)寄存器地址映射

  • 在中斷服務(wù)程序中修改全局變量

  • 在多線程、多任務(wù)應(yīng)用中,全局變量被多個任務(wù)讀寫


接下來,我們將分別討論上述三種情況。


外設(shè)寄存器


嵌入式系統(tǒng)包含真正的硬件,通常會有復(fù)雜的外設(shè)。這些外設(shè)寄存器的值可能被異步的修改。舉個簡單的例子,我們要把一個8-bit狀態(tài)寄存器的地址映射到0x1234。在程序中循環(huán)查看該狀態(tài)寄存器的值是否變?yōu)榉?。


下面是最容易想到,但錯誤的實(shí)現(xiàn)方法:

?

書本里不會講的C語言關(guān)鍵詞volatile用法


當(dāng)你打開編譯器優(yōu)化時,程序總是執(zhí)行失敗。因?yàn)榫幾g器會生成下面的匯編代碼:

?

書本里不會講的C語言關(guān)鍵詞volatile用法


程序被優(yōu)化的原因很簡單,既然已經(jīng)把變量的值讀入累加器,就沒有必要重新一遍,編譯器認(rèn)為值是不會變化的。就這樣,在第三行,程序進(jìn)入了無限死循環(huán)。為了告訴編譯器我們的真正意圖,我們需要修改函數(shù)的聲明:

?

書本里不會講的C語言關(guān)鍵詞volatile用法


編譯器生成的匯編代碼:

?

書本里不會講的C語言關(guān)鍵詞volatile用法


像這樣,我們得到了正確的動作。


中斷服務(wù)程序


在中斷服務(wù)程序中,經(jīng)常會修改一些全局變量值,來作為主程序中的判斷條件。例如,在串口中斷服務(wù)程序中,可能會檢測是否接收到了ETX(假如是消息的結(jié)束標(biāo)識符)字符。如果接收到了ETX,ISR設(shè)置一個全局標(biāo)志位。


錯誤的做法:

?

書本里不會講的C語言關(guān)鍵詞volatile用法


在關(guān)閉編譯器優(yōu)化的情況下,程序可能執(zhí)行正常。然而,任何像樣點(diǎn)而優(yōu)化都會“break”這段程序。問題是編譯器并不知道etx_rcvd可能被ISR中被修改。編譯器只知道,表達(dá)式!ext_rcvd始終為真,你講用于無法退出循環(huán)。結(jié)果,循環(huán)后面的代碼可能被編譯器優(yōu)化掉。


幸運(yùn)的話,你的編譯器可能會發(fā)出警告;不幸的話,(或者你不認(rèn)真的查看編譯器警告),你的程序無法正常執(zhí)行。當(dāng)然,你可以責(zé)怪編譯器執(zhí)行了“糟糕的優(yōu)化”。


解決方式是,將變量etx_rcvd聲明為volatile,所有問題(當(dāng)然,也可能是部分問題)就消失了。


多線程應(yīng)用


在實(shí)時系統(tǒng)中,盡管有想queues,pipes等這些同步機(jī)制,使用全局變量實(shí)現(xiàn)兩個任務(wù)共享信息的做法依然很常見。即使在你的程序中加入了搶占式調(diào)度器,你的編譯器依然無法知道什么是上下文切換,或何時發(fā)生上下文切換。因此從概念上講,多任務(wù)修改全局變量的的做法與中斷服務(wù)程序中修改全局變量的做法是相同的。因此,所有這類全局變量都應(yīng)該聲明為volatile。例如下面的程序:

?

書本里不會講的C語言關(guān)鍵詞volatile用法


當(dāng)打開編譯器優(yōu)化時,這段程序可能執(zhí)行失敗。解決方法是將cntr聲明為volatile。


書本里不會講的C語言關(guān)鍵詞volatile用法最后的思考


一些編譯器允許你把所有的變量隱式的聲明為volatile。請抵制這種誘惑,因?yàn)樗鼤钅悴辉偎伎?,?dāng)然也會導(dǎo)致生成低效的代碼。


另外,也不要責(zé)怪優(yōu)化器或直接把它關(guān)掉。現(xiàn)代的優(yōu)化器已經(jīng)足夠優(yōu)秀,我已經(jīng)記不清上次遇到優(yōu)化bug是什么時候了。相反,我常??吹匠绦騿T們錯誤的使用volatile。


如果你被要求去修改一個很古怪的代碼,請在程序中查找一下volatile關(guān)鍵字;如果你什么也沒有找到,上面討論的例子可以向你提供一些解決問題的思路。



來源:網(wǎng)絡(luò)版權(quán)歸原作者所有,如有侵權(quán),請聯(lián)系刪除。

21ic電子網(wǎng)

掃描二維碼,關(guān)注更多精彩內(nèi)容

本站聲明: 本文章由作者或相關(guān)機(jī)構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點(diǎn),本站亦不保證或承諾內(nèi)容真實(shí)性等。需要轉(zhuǎn)載請聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請及時聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車的華為或?qū)⒋呱龈蟮莫?dú)角獸公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關(guān)鍵字: 阿維塔 塞力斯 華為

加利福尼亞州圣克拉拉縣2024年8月30日 /美通社/ -- 數(shù)字化轉(zhuǎn)型技術(shù)解決方案公司Trianz今天宣布,該公司與Amazon Web Services (AWS)簽訂了...

關(guān)鍵字: AWS AN BSP 數(shù)字化

倫敦2024年8月29日 /美通社/ -- 英國汽車技術(shù)公司SODA.Auto推出其旗艦產(chǎn)品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認(rèn)證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開發(fā)耗時1.5...

關(guān)鍵字: 汽車 人工智能 智能驅(qū)動 BSP

北京2024年8月28日 /美通社/ -- 越來越多用戶希望企業(yè)業(yè)務(wù)能7×24不間斷運(yùn)行,同時企業(yè)卻面臨越來越多業(yè)務(wù)中斷的風(fēng)險,如企業(yè)系統(tǒng)復(fù)雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務(wù)連續(xù)性,提升韌性,成...

關(guān)鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報道,騰訊和網(wǎng)易近期正在縮減他們對日本游戲市場的投資。

關(guān)鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會開幕式在貴陽舉行,華為董事、質(zhì)量流程IT總裁陶景文發(fā)表了演講。

關(guān)鍵字: 華為 12nm EDA 半導(dǎo)體

8月28日消息,在2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會上,華為常務(wù)董事、華為云CEO張平安發(fā)表演講稱,數(shù)字世界的話語權(quán)最終是由生態(tài)的繁榮決定的。

關(guān)鍵字: 華為 12nm 手機(jī) 衛(wèi)星通信

要點(diǎn): 有效應(yīng)對環(huán)境變化,經(jīng)營業(yè)績穩(wěn)中有升 落實(shí)提質(zhì)增效舉措,毛利潤率延續(xù)升勢 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務(wù)引領(lǐng)增長 以科技創(chuàng)新為引領(lǐng),提升企業(yè)核心競爭力 堅持高質(zhì)量發(fā)展策略,塑強(qiáng)核心競爭優(yōu)勢...

關(guān)鍵字: 通信 BSP 電信運(yùn)營商 數(shù)字經(jīng)濟(jì)

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺與中國電影電視技術(shù)學(xué)會聯(lián)合牽頭組建的NVI技術(shù)創(chuàng)新聯(lián)盟在BIRTV2024超高清全產(chǎn)業(yè)鏈發(fā)展研討會上宣布正式成立。 活動現(xiàn)場 NVI技術(shù)創(chuàng)新聯(lián)...

關(guān)鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會上,軟通動力信息技術(shù)(集團(tuán))股份有限公司(以下簡稱"軟通動力")與長三角投資(上海)有限...

關(guān)鍵字: BSP 信息技術(shù)
關(guān)閉
關(guān)閉