一篇文章帶你了解C語言中volatile關(guān)鍵字
C語言中volatile關(guān)鍵字
volatile關(guān)鍵字是C語言中非常冷門的關(guān)鍵字,因為用到這個關(guān)鍵字的場景并不多。
當不用這個關(guān)鍵字的時候,CPU可能會對我們的代碼做一定的優(yōu)化:
內(nèi)存中的數(shù)據(jù)要放入CPU中進行運算或控制,而這個數(shù)據(jù)的值是被放入寄存器中,然后再將寄存器中的數(shù)據(jù)進行運算或控制的,對于一個死循環(huán)int flag=1;while(flag);來說;如果進行優(yōu)化,則下次循環(huán)則不需要再次將flag內(nèi)存中的值放入寄存器中,而是直接使用寄存器中已有的值進行循環(huán);如果不進行優(yōu)化,則下次還需要將flag內(nèi)存中的值放入寄存器中,然后使用寄存器中的數(shù)據(jù)。
總結(jié)起來就是,遇到這個關(guān)鍵字聲明的變量,編譯器對訪問該變量的代碼就不再進行優(yōu)化,從而可以提供對其地址的穩(wěn)定訪問;如果不使用valatile,則編譯器將對所聲明的語句進行優(yōu)化。
這兩種情況在單線程的情況下區(qū)別不大,但是在多線程的情況下可能會有其他邏輯將flag修改為0,如果進行優(yōu)化,則死循環(huán)不會停下來。
所以volatile的作用就是讓變量不要被CPU優(yōu)化,達到穩(wěn)定訪問內(nèi)存的目的。
比如下面的代碼:

我們在gcc下使用命令gcc test.c -O2 -g將代碼進行優(yōu)化,然后用命令objdump -S -d a.out > a.s將優(yōu)化后的匯編代碼放入a.s文件中,再用vim a.s查看a.s文件:

程序會一直在這一句代碼中死循環(huán):

加入volatile后:

再用相同的命令查看a.s文件:

可以看到每次循環(huán)都會讀取pass的數(shù)據(jù)。
結(jié)論: volatile 忽略編譯器的優(yōu)化,保持內(nèi)存可見性。
另外,const和volatile是不沖突的:
const volatile int a = 10;
const關(guān)鍵字要求變量a不能直接被寫入,而volatile關(guān)鍵字要求每次讀取數(shù)據(jù)的時候,都要從a所在的內(nèi)存中讀取,并不會改變變量a的值。
總結(jié)
本篇文章就到這里了,希望能夠給你帶來幫助,也希望您能夠多多關(guān)注腳本之家的更多內(nèi)容!
相關(guān)文章
C++實現(xiàn)LeetCode(53.最大子數(shù)組)
這篇文章主要介紹了C++實現(xiàn)LeetCode(53.最大子數(shù)組),本篇文章通過簡要的案例,講解了該項技術(shù)的了解與使用,以下就是詳細內(nèi)容,需要的朋友可以參考下2021-07-07

