精品国产人成在线_亚洲高清无码在线观看_国产在线视频国产永久2021_国产AV综合第一页一个的一区免费影院黑人_最近中文字幕MV高清在线视频

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

那些書本上都沒有提到的C語言volatile用法

STM32嵌入式開發(fā) ? 來源:嵌入式ARM ? 作者:嵌入式ARM ? 2021-10-12 14:47 ? 次閱讀

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

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

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

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

古怪的硬件驅(qū)動

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

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

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

只要變量可能被意外的修改,就需要把該變量聲明為volatile。在實際應(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)榉?。C語言操作寄存器的手法,可以參考這篇文章:C語言操作寄存器的常見手法。

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

bdfdf05a-2b08-11ec-82a8-dac502259ad0.png

當你打開編譯器優(yōu)化時,程序總是執(zhí)行失敗。因為編譯器會生成下面的匯編代碼:

be3e0974-2b08-11ec-82a8-dac502259ad0.png

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

be89177a-2b08-11ec-82a8-dac502259ad0.png

編譯器生成的匯編代碼:

bedf406e-2b08-11ec-82a8-dac502259ad0.png

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

中斷服務(wù)程序

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

錯誤的做法:

bf0fa740-2b08-11ec-82a8-dac502259ad0.png

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

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

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

多線程應(yīng)用

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

因此,所有這類全局變量都應(yīng)該聲明為volatile。

例如下面的程序:

bf0fa740-2b08-11ec-82a8-dac502259ad0.png

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

總結(jié)

一些編譯器允許你把所有的變量隱式的聲明為volatile。請抵制這種誘惑,因為它會令你不再思考,當然也會導(dǎo)致生成低效的代碼。

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

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

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • 串口
    +關(guān)注

    關(guān)注

    14

    文章

    1533

    瀏覽量

    75451
  • 程序
    +關(guān)注

    關(guān)注

    115

    文章

    3719

    瀏覽量

    80355
  • 代碼
    +關(guān)注

    關(guān)注

    30

    文章

    4670

    瀏覽量

    67760
  • 編譯器
    +關(guān)注

    關(guān)注

    1

    文章

    1602

    瀏覽量

    48894
  • volatile
    +關(guān)注

    關(guān)注

    0

    文章

    44

    瀏覽量

    12979

原文標題:教科書沒有講的C語言volatile用法

文章出處:【微信號:c-stm32,微信公眾號:STM32嵌入式開發(fā)】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏

    評論

    相關(guān)推薦

    NA321的figure9一直都沒有實現(xiàn)放大是為什么?

    NA321的figure9有人用過嗎真的可以實現(xiàn)嗎為什么我一直都沒有實現(xiàn)放大?
    發(fā)表于 09-12 06:59

    lmh6518配置為LG模式時,無論增益配置為多少,都沒有信號輸出,為什么?

    500mv的信號,增益設(shè)置為38.8db,信號出來只有180mv,經(jīng)過兩次阻抗匹配理論增益應(yīng)該還有32.8dB。請問是技術(shù)手冊理解錯了還是可能的其他什么問題?對輔助輸出及濾波器帶寬的控制都沒有
    發(fā)表于 09-05 07:37

    iso122用于隔離交流分壓后的電壓,不管有沒有輸入,輸出都沒有變化,為什么?

    輸入是交流分壓后的電壓,2V左右,但是不管有沒有輸入都輸出一點影響都沒有,兩個+-15是共用的,一個時直流地一個時交流地,不知道是不我的接法有問題,還有應(yīng)該有其他的連接方法,自己焊接了一個電路圖也是出現(xiàn)一樣的問題,輸入對輸出沒有
    發(fā)表于 08-30 07:17

    使用pspice軟件進行仿真,始終都沒有高低電平信號出現(xiàn),為什么?

    使用pspice軟件進行仿真,如圖出現(xiàn)問題,始終都沒有高低電平信號出現(xiàn)。
    發(fā)表于 08-20 06:05

    ESP32C3燒錄fast_scan的例程代碼,怎么一點東西都沒有顯示的?

    最近研究ESP32C3的WIFI 部分,燒錄fast_scan的例程代碼,怎么一點東西都沒有顯示的?只是顯示了ESP32的MAC地址?什么東西都沒有顯示或者動作了?我是不是要在menuconfig設(shè)置某些東西才能掃描AP?還是要
    發(fā)表于 06-17 06:08

    請問stm32復(fù)位電路為什么都沒有放電二極管?

    好多資料里面,stm32復(fù)位電路為什么都沒有放電二極管?
    發(fā)表于 04-01 08:18

    STM32G030F6PCUbe生成的TIM中斷不能運行,打斷點都沒有進入是為什么?

    STM32G030F6PCUbe生成的TIM中斷不能運行,打斷點都沒有進入
    發(fā)表于 03-13 06:56

    C語言的指針用法

    C語言編程中善用指針可以簡化一些任務(wù)的處理,而對于一些任務(wù)(比如動態(tài)內(nèi)存分配),必須要有指針才行的。也就是說精通C指針編程是很有必要的,幫助你成為一名優(yōu)秀的Cer。
    發(fā)表于 03-05 14:22 ?256次閱讀
    <b class='flag-5'>C</b><b class='flag-5'>語言</b>的指針<b class='flag-5'>用法</b>

    E203不顯示pass與fail是怎么回事,.log里什么都沒有還沒報錯?

    蜂鳥E203自測試用例失敗,不顯示pass與fail怎么回事,.log里什么都沒有,還沒報錯
    發(fā)表于 01-10 07:56

    C語言-#和##的具體用法

    C語言中,在宏里面使用’#’和’##’有它非常神奇的作用。在宏定義的替換的過程中,#號可以作為一個預(yù)處理運算符,把宏參數(shù)轉(zhuǎn)換為字符串。##運算符則可以把兩個宏參數(shù)組合在一起。下面就來說說具體的用法
    的頭像 發(fā)表于 12-19 12:54 ?3291次閱讀
    <b class='flag-5'>C</b><b class='flag-5'>語言</b>-#和##的具體<b class='flag-5'>用法</b>

    C語言C++中那些不同的地方

    ++11標準。根據(jù)不同的標準,它們的功能也會有所不同,但是越新的版本支持的編譯器越少,所以本文在討論的時候使用的C語言標準是C89,C++標準是C
    的頭像 發(fā)表于 12-07 14:29 ?772次閱讀
    <b class='flag-5'>C</b><b class='flag-5'>語言</b>和<b class='flag-5'>C</b>++中<b class='flag-5'>那些</b>不同的地方

    C語言for循環(huán)的用法和注意事項

    C 語言是一種廣泛使用的編程語言,它具有簡潔、高效、靈活的特點。C 語言中有很多控制流程的語句,其中 for 循環(huán)是一種常見的循環(huán)結(jié)構(gòu),可以
    的頭像 發(fā)表于 11-20 18:27 ?1676次閱讀
    <b class='flag-5'>C</b><b class='flag-5'>語言</b>for循環(huán)的<b class='flag-5'>用法</b>和注意事項

    AD9956單端輸入和輸出的原理圖應(yīng)該怎樣設(shè)計,datasheet和評估板里都沒有參考方案?

    我們的AD9956設(shè)計成單端輸入和輸出,但是DDS沒有輸出,經(jīng)測試SYNC_OUT管教并沒有輸出,供電應(yīng)該是沒有問題的,我懷疑是原理圖有問題,想問下AD9956單端輸入和輸出的原理圖應(yīng)該怎樣設(shè)計,datasheet和評估板里
    發(fā)表于 11-16 06:47

    為什么SPI接口的液晶屏都沒有MISO線?用硬件SPI刷320*240屏速度怎么樣?

    為什么SPI接口的液晶屏都沒有MISO線? 用硬件SPI刷320*240屏速度怎么樣?
    發(fā)表于 11-08 06:48

    ld3320語音模塊控制繼電器全開之后喊口令都沒有反應(yīng)是為什么?

    我用ld3320控制繼電器,從而控制燈,風扇,窗簾的開關(guān),但是每次我全開之后ld3320就不會再識別了,我怎么喊口令都沒有反應(yīng),想問問這可能是什么原因引起的,怎樣能解決,我已經(jīng)檢查了好多天了,就是查不出原因,求大神幫幫忙?
    發(fā)表于 11-06 06:36