日本免费高清视频-国产福利视频导航-黄色在线播放国产-天天操天天操天天操天天操|www.shdianci.com

學(xué)無(wú)先后,達(dá)者為師

網(wǎng)站首頁(yè) 編程語(yǔ)言 正文

C/C++中的靜態(tài)變量注意事項(xiàng)_C 語(yǔ)言

作者:宇龍_ ? 更新時(shí)間: 2022-09-23 編程語(yǔ)言

前言

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)局部變量無(wú)論如何都只會(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è)方法都能替代。或者只使用常量去初始化靜態(tài)變量,這能保證線程安全!

總結(jié),編譯器在我們不知道的地方默默付出,大家要知道感恩!

原文鏈接:https://blog.csdn.net/Think88666/article/details/126004224

欄目分類
最近更新