chinese直男口爆体育生外卖, 99久久er热在这里只有精品99, 又色又爽又黄18禁美女裸身无遮挡, gogogo高清免费观看日本电视,私密按摩师高清版在线,人妻视频毛茸茸,91论坛 兴趣闲谈,欧美 亚洲 精品 8区,国产精品久久久久精品免费

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

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

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

那些書(shū)本上都沒(méi)有提到的C語(yǔ)言volatile用法

STM32嵌入式開(kāi)發(fā) ? 來(lái)源:嵌入式ARM ? 作者:嵌入式ARM ? 2021-10-12 14:47 ? 次閱讀
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

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

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

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

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

古怪的硬件驅(qū)動(dòng)

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

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

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

C語(yǔ)言關(guān)鍵字volatile語(yǔ)法

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

volatile int foo;int volatile foo;

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

volatile uint8_t * pReg;uint8_t volatile * pReg;

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

int * volatile p;

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

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

正確使用C語(yǔ)言關(guān)鍵字volatile

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

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

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

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

接下來(lái),我們將分別討論上述三種情況。

外設(shè)寄存器

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

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

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

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

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

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

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

編譯器生成的匯編代碼:

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

像這樣,我們得到了正確的動(dòng)作。

中斷服務(wù)程序

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

錯(cuò)誤的做法:

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

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

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

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

多線程應(yīng)用

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

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

例如下面的程序:

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

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

總結(jié)

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

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

如果你被要求去修改一個(gè)很古怪的代碼,請(qǐng)?jiān)诔绦蛑胁檎乙幌聉olatile關(guān)鍵字;如果你什么也沒(méi)有找到,上面討論的例子可以向你提供一些解決問(wèn)題的思路。
編輯:jq

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

    關(guān)注

    15

    文章

    1605

    瀏覽量

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

    關(guān)注

    117

    文章

    3836

    瀏覽量

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

    關(guān)注

    30

    文章

    4940

    瀏覽量

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

    關(guān)注

    1

    文章

    1669

    瀏覽量

    51059
  • volatile
    +關(guān)注

    關(guān)注

    0

    文章

    46

    瀏覽量

    13616

原文標(biāo)題:教科書(shū)沒(méi)有講的C語(yǔ)言volatile用法

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

收藏 人收藏
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

    評(píng)論

    相關(guān)推薦
    熱點(diǎn)推薦

    什么是??volatile

    volatile是一個(gè)類(lèi)型修飾符(type specifier)。 volatile的作用是作為指令關(guān)鍵字,確保本條指令不會(huì)因編譯器的優(yōu)化而省略,且要求每次直接讀值。 volatile變量是說(shuō)這變量可能會(huì)被意想不到地改變,這樣
    發(fā)表于 11-25 06:36

    C語(yǔ)言和單片機(jī)C語(yǔ)言有什么差異

    匯編語(yǔ)言機(jī)器才能讀懂,所以每個(gè)平臺(tái)的編譯器編譯成對(duì)應(yīng)平臺(tái)匯編的程序,每個(gè)平臺(tái)的匯編不一樣,當(dāng)然編譯器也不一樣。 DOS的TC2 TC3 WINDOWS的VC 8051的C51都有自
    發(fā)表于 11-14 07:55

    C語(yǔ)言的printf基本用法介紹

    個(gè)簡(jiǎn)單的例子: printf(\"C語(yǔ)言\"); 這個(gè)語(yǔ)句可以在屏幕顯示“C語(yǔ)言”,與puts(\"
    發(fā)表于 11-12 07:04

    k230接入規(guī)定電源后,只亮紅燈,按鍵都沒(méi)有反應(yīng),電腦識(shí)別不出sdcard,請(qǐng)問(wèn)是燒壞了嗎?

    k230接入規(guī)定電源后,只亮紅燈,按鍵都沒(méi)有反應(yīng),電腦識(shí)別不出sdcard,請(qǐng)問(wèn)是燒壞了嗎
    發(fā)表于 07-29 12:25

    提高篇——C語(yǔ)言核心技術(shù)(中文版)

    該資料是“C編程語(yǔ)言”和“C語(yǔ)言鏈接庫(kù)”的完整參考手冊(cè)。這本書(shū)的目的是提供一本方便、可靠的手冊(cè),輔助日常的編程工作。本書(shū)描述C
    發(fā)表于 06-13 16:39

    如何在 樹(shù)莓派 編寫(xiě)和運(yùn)行 C 語(yǔ)言程序?

    在本教程中,我將討論C編程語(yǔ)言是什么,C編程的用途,以及如何在RaspberryPi編寫(xiě)和運(yùn)行C程序。本文的目的是為您介紹在Raspber
    的頭像 發(fā)表于 03-25 09:28 ?927次閱讀
    如何在 樹(shù)莓派 <b class='flag-5'>上</b>編寫(xiě)和運(yùn)行 <b class='flag-5'>C</b> <b class='flag-5'>語(yǔ)言</b>程序?

    六腳IC:有知道這個(gè)六腳IC的嗎,找過(guò)好多地方都沒(méi)有找到?

    有知道這個(gè)六腳IC的嗎,找過(guò)好多地方都沒(méi)有找到。拜托拜托!?。?
    發(fā)表于 03-21 15:18

    FreeRTOS(V9.0)中創(chuàng)建信號(hào)量的函數(shù)都沒(méi)有被定義,因此用不了,怎么解決

    問(wèn)題背景:我想要使用信號(hào)量,結(jié)果查找了整個(gè)工程都沒(méi)有創(chuàng)建信號(hào)量的函數(shù)。我還以為是我自己移植有問(wèn)題,因此還特地下載了其他人移植好的工程進(jìn)行編程。結(jié)果也沒(méi)有創(chuàng)建信號(hào)量的函數(shù)。不論是二值信號(hào)量創(chuàng)建函數(shù)
    發(fā)表于 03-13 09:30

    ADS1251的DOUT引腳,開(kāi)始是有波形,但是度數(shù)卻全部為1,后來(lái)波形都沒(méi)有了一直為高是怎么回事?

    ADS1251的DOUT引腳,開(kāi)始是有波形,但是度數(shù)卻全部為1,后來(lái)波形都沒(méi)有了,一直為高,求解答!
    發(fā)表于 01-15 07:08

    ADS1278電后運(yùn)行一段時(shí)間后DRDY信號(hào)就沒(méi)有了,為什么?

    。沒(méi)有辦法,我們重新焊接了3塊樣板,但這 3 塊怎么都沒(méi)有 DRDY 信號(hào)出現(xiàn),后來(lái)我們將第一塊運(yùn)行一段時(shí)間 DRDY 信號(hào)丟失的那個(gè) ADS1278 芯片和后來(lái)的樣板的 ADS1278 對(duì)調(diào),但對(duì)調(diào)后
    發(fā)表于 01-09 07:55

    EE-192:使用C語(yǔ)言在Blackfin處理器創(chuàng)建中斷驅(qū)動(dòng)系統(tǒng)

    電子發(fā)燒友網(wǎng)站提供《EE-192:使用C語(yǔ)言在Blackfin處理器創(chuàng)建中斷驅(qū)動(dòng)系統(tǒng).pdf》資料免費(fèi)下載
    發(fā)表于 01-03 15:03 ?0次下載
    EE-192:使用<b class='flag-5'>C</b><b class='flag-5'>語(yǔ)言</b>在Blackfin處理器<b class='flag-5'>上</b>創(chuàng)建中斷驅(qū)動(dòng)系統(tǒng)

    調(diào)C6748讀ADS1271,通過(guò)MCBSP就是讀不到,連DRDY型號(hào)都沒(méi)有,為什么?

    。實(shí)驗(yàn)表明,是沒(méi)有的,我想不明白為什么沒(méi)有?請(qǐng)大師指點(diǎn),現(xiàn)在在調(diào)C6748讀ADS1271,通過(guò)數(shù)據(jù)模擬的方式可以讀取到數(shù)據(jù),但通過(guò)MCBSP就是讀不到,連DRDY型號(hào)都沒(méi)有
    發(fā)表于 01-01 06:59

    MSP430f5529一直都沒(méi)有輸出是怎么回事?

    用普通io控制的cs,sclk,din,芯片用的是MSP430f5529,一直都沒(méi)有輸出,求助
    發(fā)表于 12-24 07:00

    按照手冊(cè)的電路圖繪制的,輸出的DAC CODE怎么調(diào)整out輸出都沒(méi)有變化,是哪里的問(wèn)題?

    請(qǐng)問(wèn)一下,我按照手冊(cè)的電路圖繪制的,輸出的DACCODE怎么調(diào)整out輸出都沒(méi)有變化,時(shí)間運(yùn)行長(zhǎng)了以后就會(huì)燒毀芯片,電源與地短路,想問(wèn)一下這個(gè)問(wèn)題怎么解決
    發(fā)表于 12-20 07:43

    STM32帶DA功能的MCU,配合XTR111應(yīng)該輸出一個(gè)4-20mA的電流,無(wú)論如何都沒(méi)有電流產(chǎn)生,為什么?

    我用的是STM32帶DA功能的MCU,現(xiàn)在配合XTR111應(yīng)該輸出一個(gè)4-20mA的電流 但是無(wú)論如何都沒(méi)有電流產(chǎn)生,我用一個(gè)獨(dú)立的XTR111模塊電路外接就可以正常使用, 那么也就是說(shuō),MCU驅(qū)動(dòng)電壓轉(zhuǎn)電流的理論是行的通的,上傳原理圖,還請(qǐng)各位幫忙指點(diǎn)一下
    發(fā)表于 12-11 08:24