C/C++中的靜態(tài)變量注意事項
前言
C/C++中的靜態(tài)變量,相信大多數(shù)人都用過,但你很可能用錯了,包括你現(xiàn)在所在的項目中都可能埋著這個坑,不信我們往下看!
正文
我們先來看一段大家常寫的代碼,很簡單,這段代碼沒啥坑:
#include <stdio.h> int GetData() { static int a = 0; return a++; } int main() { for (int i = 0; i < 100; ++i) { printf("%d\n", GetData()); } }
大家都清楚,靜態(tài)變量只初始化一次,所以GetData調(diào)用了100次,打印的結(jié)果也是0-99,想必大家都很清楚 ,那請問GetData中初始化變量a的代碼只會執(zhí)行一次?是在哪個階段初始靜態(tài)局部變量a的?
想必大家都能回答上來,靜態(tài)局部變量a的生命周期從程序運(yùn)行開始就已經(jīng)存在并初始化了的,并非是在GetData函數(shù)中初始化的,但又不完全對,我們看下一段代碼:
#include <stdio.h> int GetA() { return 0; } int GetData() { static int a = GetA(); return a++; } int main() { for (int i = 0; i < 100; ++i) { printf("%d\n", GetData()); } }
看了這段代碼,不知道大家有沒有懵逼?問題來了,請問GetA函數(shù)會被調(diào)用幾次?靜態(tài)局部變量a是在什么時候初始化的?給大家5秒鐘思考!
OK!靜態(tài)局部變量無論如何都只會初始化一次,這是沒有毛病的,但此時靜態(tài)局部變量a是在第一次調(diào)用GetData函數(shù)的時候才被初始化的,與前一個例子用常量初始化靜態(tài)變量并不相同,當(dāng)然生命周期還是從程序運(yùn)行開始到程序結(jié)束為止。
那編譯器是怎么初始化靜態(tài)變量a的呢?編譯器會改造GetData方法如下:
int GetData() { static bool init = false; if (!init) { a = GetA();//a已經(jīng)被定義在全局了 init = true; } return a++; }
這樣編譯器就可以保證靜態(tài)變量a在GetData函數(shù)內(nèi)只被初始化一次,但請問a的初始化是否線程安全?
當(dāng)然,不同編譯器的實(shí)現(xiàn)并不相同,有的編譯器會在初始化全局變量a的時候用上臨界區(qū)等,以保證初始化的線程安全,有的卻并沒有,當(dāng)然為了自己的代碼兼容性更強(qiáng),建議不要這樣寫,隨便換個方法都能替代?;蛘咧皇褂贸A咳コ跏蓟o態(tài)變量,這能保證線程安全!
總結(jié),編譯器在我們不知道的地方默默付出,大家要知道感恩!
到此這篇關(guān)于C/C++中的靜態(tài)變量注意事項的文章就介紹到這了,更多相關(guān)C++ 靜態(tài)變量內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C語言中sizeof()與strlen()函數(shù)的使用入門及對比
這篇文章主要介紹了C語言中sizeof()與strlen()函數(shù)的使用入門及對比,同時二者在C++中的使用情況也基本上同理,是需要的朋友可以參考下2015-12-12C語言完整實(shí)現(xiàn)12種排序算法(小結(jié))
本文主要介紹了C語言完整實(shí)現(xiàn)12種排序算法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-05-05QT通過C++線程池運(yùn)行Lambda自定義函數(shù)流程詳解
最近在接觸公司的一個QT桌面項目,其中里面有一個模塊是使用線程池去運(yùn)行自定義函數(shù)的,自己潛心研究那個線程池代碼一天,發(fā)現(xiàn)研究不透,看不懂,里面幾乎都是使用C++11的新特性進(jìn)行編寫2022-10-10基于C++的農(nóng)夫過河問題算法設(shè)計與實(shí)現(xiàn)方法
這篇文章主要介紹了基于C++的農(nóng)夫過河問題算法設(shè)計與實(shí)現(xiàn)方法,簡單描述了農(nóng)夫過河問題,并結(jié)合實(shí)例形式詳細(xì)分析了基于C++實(shí)現(xiàn)農(nóng)夫過河問題的相關(guān)算法實(shí)現(xiàn)步驟與操作技巧,需要的朋友可以參考下2017-09-09c++ priority_queue用法入門超詳細(xì)教程
priority_queue即優(yōu)先級隊列,它的使用場景很多,它底層是用大小根堆實(shí)現(xiàn)的,可以用log(n)的時間動態(tài)地維護(hù)數(shù)據(jù)的有序性,這篇文章主要介紹了c++ priority_queue用法入門超詳細(xì)教程,需要的朋友可以參考下2023-12-12