国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

詳解C中volatile關鍵字

開發 后端
volatile關鍵字是一種類型修飾符,用它聲明的類型變量表示可以被某些編譯器未知的因素更改,遇到這個關鍵字聲明的變量,編譯器對訪問該變量的代碼就不再進行優化,從而可以提供對特殊地址的穩定訪問。 來看本文。

volatile是一個類型修飾符(type specifier),它是被設計用來修飾被不同線程訪問和修改的變量。如果沒有volatile,基本上會導致這樣的結果:要么無法編寫多線程程序,要么編譯器失去大量優化的機會。

volatile提醒編譯器它后面所定義的變量隨時都有可能改變,因此編譯后的程序每次需要存儲或讀取這個變量的時候,都會直接從變量地址中讀取數據。如果沒有volatile關鍵字,則編譯器可能優化讀取和存儲,可能暫時使用寄存器中的值,如果這個變量由別的程序更新了的話,將出現不一致的現象。下面舉例說明。在DSP開發中,經常需要等待某個事件的觸發,所以經常會寫出這樣的程序:

  1. short flag;  
  2. void test()  
  3. {  
  4. do1();  
  5. while(flag==0);  
  6. do2();  

這段程序等待內存變量flag的值變為1(懷疑此處是0,有點疑問,)之后才運行do2()。變量flag的值由別的程序更改,這個程序可能是某個硬件中斷服務程序。例如:如果某個按鈕按下的話,就會對DSP產生中斷,在按鍵中斷程序中修改flag為1,這樣上面的程序就能夠得以繼續運行。但是,編譯器并不知道flag的值會被別的程序修改,因此在它進行優化的時候,可能會把flag的值先讀入某個寄存器,然后等待那個寄存器變為1。如果不幸進行了這樣的優化,那么while循環就變成了死循環,因為寄存器的內容不可能被中斷服務程序修改。為了讓程序每次都讀取真正flag變量的值,就需要定義為如下形式:

  1. volatile short flag; 

需要注意的是,沒有volatile也可能能正常運行,但是可能修改了編譯器的優化級別之后就又不能正常運行了。因此經常會出現debug版本正常,但是release版本卻不能正常的問題。所以為了安全起見,只要是等待別的程序修改某個變量的話,就加上volatile關鍵字。

一、volatile的本意是“易變的”

由于訪問寄存器的速度要快過RAM,所以編譯器一般都會作減少存取外部RAM的優化。比如:

  1. static int i=0;  
  2. int main(void)  
  3. {  
  4. ...  
  5. while (1)  
  6. {  
  7. if (i) do_something();  
  8. }  
  9. }  
  10. /* Interrupt service routine. */ 
  11. void ISR_2(void)  
  12. {  
  13. i=1;  

程序的本意是希望ISR_2中斷產生時,在main當中調用do_something函數,但是,由于編譯器判斷在main函數里面沒有修改過i,因此可能只執行一次對從i到某寄存器的讀操作,然后每次if判斷都只使用這個寄存器里面的“i副本”,導致do_something永遠也不會被調用。如果變量加上volatile修飾,則編譯器保證對此變量的讀寫操作都不會被優化(肯定執行)。此例中i也應該如此說明。

一般說來,volatile用在如下的幾個地方:

1、中斷服務程序中修改的供其它程序檢測的變量需要加volatile;

2、多任務環境下各任務間共享的標志應該加volatile;

3、存儲器映射的硬件寄存器通常也要加volatile說明,因為每次對它的讀寫都可能由不同意義;

另外,以上這幾種情況經常還要同時考慮數據的完整性(相互關聯的幾個標志讀了一半被打斷了重寫),在1中可以通過關中斷來實現,2中可以禁止任務調度,3中則只能依靠硬件的良好設計了。

二、volatile 的含義

volatile總是與優化有關,編譯器有一種技術叫做數據流分析,分析程序中的變量在哪里賦值、在哪里使用、在哪里失效,分析結果可以用于常量合并,常量傳播等優化,進一步可以死代碼消除。但有時這些優化不是程序所需要的,這時可以用volatile關鍵字禁止做這些優化,volatile的字面含義是易變的,它有下面的作用:

1 不會在兩個操作之間把volatile變量緩存在寄存器中。在多任務、中斷、甚至setjmp環境下,變量可能被其他的程序改變,編譯器自己無法知道,volatile就是告訴編譯器這種情況。

2 不做常量合并、常量傳播等優化,所以像下面的代碼:

  1. volatile int i = 1;   
  2. if (i > 0) ...  

if的條件不會當作無條件真。

3 對volatile變量的讀寫不會被優化掉。如果你對一個變量賦值但后面沒用到,編譯器常常可以省略那個賦值操作,然而對Memory Mapped IO的處理是不能這樣優化的。

前面有人說volatile可以保證對內存操作的原子性,這種說法不大準確,其一,x86需要LOCK前綴才能在SMP下保證原子性,其二,RISC根本不能對內存直接運算,要保證原子性得用別的方法,如atomic_inc。

對于jiffies,它已經聲明為volatile變量,我認為直接用jiffies++就可以了,沒必要用那種復雜的形式,因為那樣也不能保證原子性。

你可能不知道在Pentium及后續CPU中,下面兩組指令

  1. inc jiffies   
  2. ;;   
  3. mov jiffies, %eax   
  4. inc %eax   
  5. mov %eax, jiffies  

作用相同,但一條指令反而不如三條指令快。

三、編譯器優化 → C關鍵字volatile → memory破壞描述符zz

“memory”比較特殊,可能是內嵌匯編中最難懂部分。為解釋清楚它,先介紹一下編譯器的優化知識,再看C關鍵字volatile。最后去看該描述符。 

1、編譯器優化介紹

內存訪問速度遠不及CPU處理速度,為提高機器整體性能,在硬件上引入硬件高速緩存Cache,加速對內存的訪問。另外在現代CPU中指令的執行并不一定嚴格按照順序執行,沒有相關性的指令可以亂序執行,以充分利用CPU的指令流水線,提高執行速度。以上是硬件級別的優化。再看軟件一級的優化:一種是在編寫代碼時由程序員優化,另一種是由編譯器進行優化。編譯器優化常用的方法有:將內存變量緩存到寄存器;調整指令順序充分利用CPU指令流水線,常見的是重新排序讀寫指令。對常規內存進行優化的時候,這些優化是透明的,而且效率很好。由編譯器優化或者硬件重新排序引起的問題的解決辦法是在從硬件(或者其他處理器)的角度看必須以特定順序執行的操作之間設置內存屏障(memory barrier),linux 提供了一個宏解決編譯器的執行順序問題。

  1. void Barrier(void)  

這個函數通知編譯器插入一個內存屏障,但對硬件無效,編譯后的代碼會把當前CPU寄存器中的所有修改過的數值存入內存,需要這些數據的時候再重新從內存中讀出。

2、C語言關鍵字volatile

C語言關鍵字volatile(注意它是用來修飾變量而不是上面介紹的__volatile__)表明某個變量的值可能在外部被改變,因此對這些變量的存取不能緩存到寄存器,每次使用時需要重新存取。該關鍵字在多線程環境下經常使用,因為在編寫多線程的程序時,同一個變量可能被多個線程修改,而程序通過該變量同步各個線程,例如:

  1. DWORD __stdcall threadFunc(LPVOID signal)   
  2. {   
  3. int* intSignal=reinterpret_cast<int*>(signal);   
  4. *intSignal=2;   
  5. while(*intSignal!=1)   
  6. sleep(1000);   
  7. return 0;   
  8. }  

該線程啟動時將intSignal 置為2,然后循環等待直到intSignal 為1 時退出。顯然intSignal的值必須在外部被改變,否則該線程不會退出。但是實際運行的時候該線程卻不會退出,即使在外部將它的值改為1,看一下對應的偽匯編代碼就明白了:

  1. mov ax,signal   
  2. label:   
  3. if(ax!=1)   
  4. goto label  

對于C編譯器來說,它并不知道這個值會被其他線程修改。自然就把它cache在寄存器里面。記住,C 編譯器是沒有線程概念的!這時候就需要用到volatile。volatile 的本意是指:這個值可能會在當前線程外部被改變。也就是說,我們要在threadFunc中的intSignal前面加上volatile關鍵字,這時候,編譯器知道該變量的值會在外部改變,因此每次訪問該變量時會重新讀取,所作的循環變為如下面偽碼所示:

  1. label:   
  2. mov ax,signal   
  3. if(ax!=1)   
  4. goto label  

3、Memory

有了上面的知識就不難理解Memory修改描述符了,Memory描述符告知GCC:

1)不要將該段內嵌匯編指令與前面的指令重新排序;也就是在執行內嵌匯編代碼之前,它前面的指令都執行完畢

2)不要將變量緩存到寄存器,因為這段代碼可能會用到內存變量,而這些內存變量會以不可預知的方式發生改變,因此GCC插入必要的代碼先將緩存到寄存器的變量值寫回內存,如果后面又訪問這些變量,需要重新訪問內存。

如果匯編指令修改了內存,但是GCC 本身卻察覺不到,因為在輸出部分沒有描述,此時就需要在修改描述部分增加“memory”,告訴GCC 內存已經被修改,GCC 得知這個信息后,就會在這段指令之前,插入必要的指令將前面因為優化Cache 到寄存器中的變量值先寫回內存,如果以后又要使用這些變量再重新讀取。

使用“volatile”也可以達到這個目的,但是我們在每個變量前增加該關鍵字,不如使用“memory”方便。

【編輯推薦】

  1. C++和Java 的缺省初始化問題
  2. C++設計目標及原則
  3. C/C++是程序員必須掌握的語言嗎?
  4. VC++獲得當前系統時間的幾種方案
  5. 影響C++/C程序的幾大要素
責任編輯:于鐵 來源: 博客園
相關推薦

2025-06-13 08:00:00

Java并發編程volatile

2011-06-21 09:50:51

volatile

2009-09-02 09:24:03

C# this關鍵字

2022-06-29 08:05:25

Volatile關鍵字類型

2024-01-15 10:41:31

C++關鍵字開發

2023-09-22 22:27:54

autoC++11

2025-09-15 02:00:00

2024-03-15 11:52:03

C++關鍵字編程

2010-02-05 15:51:06

C++ explici

2025-07-22 01:55:00

2019-09-04 14:14:52

Java編程數據

2022-08-17 07:53:10

Volatile關鍵字原子性

2011-07-14 23:14:42

C++static

2023-11-19 22:52:42

2013-01-30 10:12:14

Pythonyield

2010-01-26 14:35:11

C++關鍵字

2009-08-21 14:58:56

C# this關鍵字

2023-06-26 08:02:34

JSR重排序volatile

2009-06-29 18:14:23

Java多線程volatile關鍵字

2024-02-21 20:46:48

C++編程volatile
點贊
收藏

51CTO技術棧公眾號

日韩美女视频在线| 国产一区在线不卡| 精品三级在线观看| 三级黄色网址| 99久久777色| 在线观看成人一级片| 99在线观看免费视频精品观看| 欧洲成人免费视频| 开心激情综合| 欧美极品xxxx| 91麻豆精品一二三区在线| 亚洲国产精品大全| aa级大片免费在线观看| 日韩欧美一级精品久久| 黄色片网站在线观看| 欧美视频二区36p| 在线免费黄色毛片| 亚洲一区二区三区三| 日本成在线观看| 综合久久给合久久狠狠狠97色| 欧美大片在线播放| 成人一区二区三区视频| 99久久免费观看| 成人精品小蝌蚪| 欧美日韩激情视频在线观看| 99re这里只有精品首页| 免费欧美一级视频| 国产亚洲精品bt天堂精选| 成人中文字幕av| 中文成人综合网| 色久视频在线观看| 日韩欧美国产成人| 春暖花开成人亚洲区| 欧洲生活片亚洲生活在线观看| 丁香在线视频| 6080日韩午夜伦伦午夜伦| h片在线播放| 亚洲国产成人久久综合一区| free欧美| 1769国内精品视频在线播放| 国际精品欧美精品| 99一区二区| 免费在线观看日韩欧美| 日本香蕉视频在线观看| 国产欧美一二三区| 一起操在线观看| 精品日本一线二线三线不卡| 日韩天堂在线| 欧美最顶级丰满的aⅴ艳星| 综合激情一区| 在线一区高清| 国产亚洲欧洲997久久综合| 黄网免费视频| 一本高清dvd不卡在线观看| 毛片在线不卡| 亚洲日韩欧美视频一区| 麻豆国产欧美一区二区三区r| 51久久精品夜色国产麻豆| 午夜国产欧美理论在线播放| 亚洲日本japanese丝袜| 国产精品污网站| 美女毛片在线看| 亚洲视频在线观看视频| 青青视频一区二区| 蜜臀久久久99精品久久久久久| 一级网站免费观看| 日日夜夜精品免费视频| 一区二区三区小说| 人成免费电影一二三区在线观看| 在线电影一区二区三区| 免费一级欧美在线观看视频| 日本成人激情视频| 久久亚洲图片| av污在线观看| 欧美日韩www| 亚洲精品v亚洲精品v日韩精品| 91麻豆国产精品| 国产精品综合在线视频| 天天噜天天色| 亚洲精品福利在线| 日韩精品网站| 国产午夜大地久久| 色哟哟一区二区| 国产一区一区| 欧洲一区二区在线| 一区二区三区免费网站| 二区三区不卡| 91在线视频免费| 风间由美一区二区三区在线观看| 一个人免费视频www在线观看| 色久欧美在线视频观看| 伊人成年综合电影网| 日韩爱爱小视频| 亚洲女人被黑人巨大进入al| 欧美日韩国产综合网| 不卡的av中文字幕| 日韩成人免费视频| 欧美三级第一页| 国产天堂在线观看| 久久艳片www.17c.com| 日韩中文字幕区一区有砖一区| 国精产品999国精产品官网| 亚洲人成电影在线播放| 国产精品豆花视频| 成人在线看片网站| 深夜福利91大全| 日韩国产成人精品| 国产中文在线| 国产精品色午夜在线观看| 26uuu久久天堂性欧美| 午夜小视频福利在线观看| 国产精品自拍小视频| 国产亚洲欧美激情| 日韩成人免费av| 国产精品12p| 日韩免费看网站| 国语精品一区| 小明精品国产一区二区三区| 97色在线观看| 99精品一区二区| 色婷婷综合久久久中字幕精品久久| 国产在线精品一区二区三区》| 五月天久久比比资源色| 色综合久久中文| 成人eeuss影院在线观看| 久久国产精品久久久| 成人av片在线观看| 成人福利av| 91精品国产吴梦梦| 国产视频在线观看一区二区| 日本女人一区二区三区| 在线观看h网| 日韩欧美精品一区二区| 欧美三区在线观看| 日韩一区二区久久| 国产小视频在线| 精品国产91亚洲一区二区三区www 精品国产_亚洲人成在线 | av资源网站在线观看| 91亚洲精品在线| 欧美午夜xxx| 午夜精品免费| jizz在线观看视频| 欧美日韩国产不卡在线看| 51午夜精品国产| 捆绑变态av一区二区三区| 在线免费看h| 国产老熟妇精品观看| 欧美激情久久久| 亚洲夂夂婷婷色拍ww47| 外国成人激情视频| 在线看黄色av| 这里只有精品66| 精品国产网站地址| 国产精品不卡一区二区三区| 妖精视频一区二区三区| 在线观看黄色片| 日本日本精品二区免费| 亚洲人成电影在线| 中文字幕精品三区| 五月开心六月丁香综合色啪| 麻豆av在线免费看| 成人在线观看毛片| 欧美xxxx做受欧美.88| 最新国产成人在线观看| 午夜欧美精品久久久久久久| 国产网红女主播精品视频| 欧美 日韩 亚洲 一区| 国产成人jvid在线播放| 欧美色偷偷大香| 国产**成人网毛片九色| 亚洲制服一区| av在线官网| 97在线播放视频| 成人免费激情视频| 亚洲电影免费观看高清| 国产日韩欧美亚洲| 欧美激情 亚洲a∨综合| 中文日产幕无线码一区二区| 1024欧美极品| 日本日本精品二区免费| 九九热视频这里只有精品| 欧美日韩中文在线观看| 国产精品婷婷| 欧美a级一区二区| 成人丝袜18视频在线观看| 全部av―极品视觉盛宴亚洲| 蜜桃av一区二区三区电影| 99在线精品一区二区三区| 婷婷综合另类小说色区| 色综合视频在线观看| 欧美性生活一区| 亚洲视屏在线播放| 国产精品免费视频久久久| 中文字幕综合在线观看| 瑟瑟在线观看| 成人免费网址| 国产成人77亚洲精品www| 第一会所亚洲原创| 99国产欧美另类久久久精品| 精品久久久久久中文字幕|