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