国产精品久久久aaaa,日日干夜夜操天天插,亚洲乱熟女香蕉一区二区三区少妇,99精品国产高清一区二区三区,国产成人精品一区二区色戒,久久久国产精品成人免费,亚洲精品毛片久久久久,99久久婷婷国产综合精品电影,国产一区二区三区任你鲁

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

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

3天內不再提示

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

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

掃碼添加小助手

加入工程師交流群

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

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

代碼執行正常–直到你打開了編譯器優化

代碼執行正常–直到打開了中斷

古怪的硬件驅動

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

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

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

C語言關鍵字volatile語法

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

volatile int foo;int volatile foo;

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

volatile uint8_t * pReg;uint8_t volatile * pReg;

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

int * volatile p;

順便提一下,關于為什么要在數據類型前使用volatile關鍵字,請自行百度搜素。

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

正確使用C語言關鍵字volatile

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

外設寄存器地址映射

在中斷服務程序中修改全局變量

在多線程、多任務應用中,全局變量被多個任務讀寫

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

外設寄存器

嵌入式系統包含真正的硬件,通常會有復雜的外設。這些外設寄存器的值可能被異步的修改。舉個簡單的例子,我們要把一個8-bit狀態寄存器的地址映射到0x1234。在程序中循環查看該狀態寄存器的值是否變為非0。C語言操作寄存器的手法,可以參考這篇文章:C語言操作寄存器的常見手法。

下面是最容易想到,但錯誤的實現方法:

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

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

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

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

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

編譯器生成的匯編代碼:

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

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

中斷服務程序

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

錯誤的做法:

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

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

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

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

多線程應用

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

因此,所有這類全局變量都應該聲明為volatile。

例如下面的程序:

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

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

總結

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

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

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

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 串口
    +關注

    關注

    15

    文章

    1618

    瀏覽量

    82809
  • 程序
    +關注

    關注

    117

    文章

    3846

    瀏覽量

    85232
  • 代碼
    +關注

    關注

    30

    文章

    4967

    瀏覽量

    73960
  • 編譯器
    +關注

    關注

    1

    文章

    1672

    瀏覽量

    51595
  • volatile
    +關注

    關注

    0

    文章

    46

    瀏覽量

    13730

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

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

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

掃碼添加小助手

加入工程師交流群

    評論

    相關推薦
    熱點推薦

    手把手教你學51單片機-C語言

    為何物。要么只是一些很簡單的小實驗,學完后也只能做到點亮個小燈、顯示個數字等,C語言指針都沒有用到過,綜合實踐的例子更是一個沒有。在學完這類教材后無法深入下去,很難跟實際項目開發銜接起
    發表于 03-05 11:47

    volatile的修飾符

    讀數據的代碼之間的代碼沒有對i進行過操作,它會自動把上次讀的數據放在k中。而不是重新從i里面讀。這樣以來,如果i是一個寄存器變量或者表示一個端口數據就容易出錯,所以說volatile可以保證對特殊地址
    發表于 01-06 07:39

    真不敢信,PCB板就挪動了一個電阻,DDR3竟神奇變好了

    DDRx調試的問題我們經常會碰到,但PCB板這個問題卻很初級,調了一周都沒有解決,沒想到最后挪動了一個電阻就好了,不信大家來看看怎么回事!
    的頭像 發表于 01-05 15:40 ?232次閱讀
    真不敢信,PCB板<b class='flag-5'>上</b>就挪動了一個電阻,DDR3竟神奇變好了

    為什么c語言沒有輸出?

    我在寫c語言程序的時候,經常在程序編譯的時候沒有輸出,經總結,發現有以下原因: 1、scanf里沒有加 。 2、邏輯問題,可能沒有輸入,
    發表于 01-05 06:40

    請問如何實現C語言訪問MCU寄存器?

    單片機的特殊功能寄存器SFR,是SRAM地址已經確定的SRAM單元,在C語言環境下對其訪問歸納起來有兩種方法。 采用標準C的強制類型轉換和指針來實現   采用標準C的強制轉換和指針的概
    發表于 12-26 07:00

    C語言C++的區別及聯系

    沒有錯。 C++一開始被本賈尼·斯特勞斯特盧普(Bjarne Stroustrup)發明時,起初被稱為“C with Classes”,即「帶類的C」。 很明顯,它是在
    發表于 12-24 07:23

    什么是??volatile

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

    C語言和單片機C語言有什么差異

    匯編語言機器才能讀懂,所以每個平臺的編譯器編譯成對應平臺匯編的程序,每個平臺的匯編不一樣,當然編譯器也不一樣。 DOS的TC2 TC3 WINDOWS的VC 8051的C51都有自
    發表于 11-14 07:55

    C語言的printf基本用法介紹

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

    k230接入規定電源后,只亮紅燈,按鍵都沒有反應,電腦識別不出sdcard,請問是燒壞了嗎?

    k230接入規定電源后,只亮紅燈,按鍵都沒有反應,電腦識別不出sdcard,請問是燒壞了嗎
    發表于 07-29 12:25

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

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

    主流的 MCU 開發語言為什么是 C 而不是 C++?

    在單片機的地界兒里,C語言穩坐中軍帳,C++想分杯羹?難嘍。咱電子工程師天天跟那針尖大的內存空間較勁,C++那些花里胡哨的玩意兒,在這兒真玩
    的頭像 發表于 05-21 10:33 ?1037次閱讀
    主流的 MCU 開發<b class='flag-5'>語言</b>為什么是 <b class='flag-5'>C</b> 而不是 <b class='flag-5'>C</b>++?

    如何在 樹莓派 編寫和運行 C 語言程序?

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

    六腳IC:有知道這個六腳IC的嗎,找過好多地方都沒有找到?

    有知道這個六腳IC的嗎,找過好多地方都沒有找到。拜托拜托!!!
    發表于 03-21 15:18

    FreeRTOS(V9.0)中創建信號量的函數都沒有被定義,因此用不了,怎么解決

    問題背景:我想要使用信號量,結果查找了整個工程都沒有創建信號量的函數。我還以為是我自己移植有問題,因此還特地下載了其他人移植好的工程進行編程。結果也沒有創建信號量的函數。不論是二值信號量創建函數
    發表于 03-13 09:30