C/C++中的靜態(tài)變量注意事項(xiàng)
前言
C/C++中的靜態(tài)變量,相信大多數(shù)人都用過,但你很可能用錯了,包括你現(xiàn)在所在的項(xiàng)目中都可能埋著這個坑,不信我們往下看!
正文
我們先來看一段大家常寫的代碼,很簡單,這段代碼沒啥坑:
#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)變量注意事項(xiàng)的文章就介紹到這了,更多相關(guān)C++ 靜態(tài)變量內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C++數(shù)據(jù)結(jié)構(gòu)之鏈表的創(chuàng)建
這篇文章主要介紹了C++數(shù)據(jù)結(jié)構(gòu)之鏈表的創(chuàng)建的相關(guān)資料,希望通過本文幫助到大家,讓大家理解掌握這部分內(nèi)容,需要的朋友可以參考下2017-10-10
C++數(shù)據(jù)模型應(yīng)用在QML委托代理機(jī)制中
這篇文章主要介紹了在QML委托代理機(jī)制中使用C++數(shù)據(jù)模型,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-08-08
C++輸入一個字符串,把其中的字符按照逆序輸出的兩種方法解析
以下是對C++中輸入一個字符串,把其中的字符按照逆序輸出的兩種方法進(jìn)行了詳細(xì)的分析介紹,需要的朋友可以過來參考下2013-07-07

