亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > C++ > 正文

基于C++全局變量的聲明與定義的詳解

2020-01-26 16:02:35
字體:
來源:轉載
供稿:網友
(1)編譯單元(模塊)
在VC或VS上編寫完代碼,點擊編譯按鈕準備生成exe文件時,編譯器做了兩步工作:
第一步,將每個.cpp(.c)和相應的.h文件編譯成obj文件;
第二步,將工程中所有的obj文件進行LINK,生成最終.exe文件。
那么,錯誤可能在兩個地方產生:
一個,編譯時的錯誤,這個主要是語法錯誤;
一個,鏈接時的錯誤,主要是重復定義變量等。
編譯單元指在編譯階段生成的每個obj文件。
一個obj文件就是一個編譯單元。
一個.cpp(.c)和它相應的.h文件共同組成了一個編譯單元。
一個工程由很多編譯單元組成,每個obj文件里包含了變量存儲的相對地址等。
(2)聲明與定義
函數或變量在聲明時,并沒有給它實際的物理內存空間,它有時候可保證你的程序編譯通過;
函數或變量在定義時,它就在內存中有了實際的物理空間。

如果你在編譯單元中引用的外部變量沒有在整個工程中任何一個地方定義的話,那么即使它在編譯時可以通過,在連接時也會報錯,因為程序在內存中找不到這個變量。
函數或變量可以聲明多次,但定義只能有一次。
(3) extern作用
作用一:
當它與"C"一起連用時,如extern "C" void fun(int a, int b);,則編譯器在編譯fun這個函數名時按C的規則去翻譯相應的函數名而不是C++的。
作用二:當它不與"C"在一起修飾變量或函數時,如在頭文件中,extern int g_nNum;,它的作用就是聲明函數或變量的作用范圍的關鍵字,其聲明的函數和變量可以在本編譯單元或其他編譯單元中使用。
即B編譯單元要引用A編譯單元中定義的全局變量或函數時,B編譯單元只要包含A編譯單元的頭文件即可,在編譯階段,B編譯單元雖然找不到該函數或變量,但它不會報錯,它會在鏈接時從A編譯單元生成的目標代碼中找到此函數。
(4)全局變量(extern)
有兩個類都需要使用共同的變量,我們將這些變量定義為全局變量。比如,res.h和res.cpp分別來聲明和定義全局變量,類ProducerThread和ConsumerThread來使用全局變量。(以下是QT工程代碼)
復制代碼 代碼如下:

/**********res.h聲明全局變量************/ 
#pragma once 

#include <QSemaphore> 

const int g_nDataSize = 1000; // 生產者生產的總數據量 
const int g_nBufferSize = 500; // 環形緩沖區的大小 

extern char g_szBuffer[]; // 環形緩沖區 
extern QSemaphore g_qsemFreeBytes; // 控制環形緩沖區的空閑區(指生產者還沒填充數據的區域,或者消費者已經讀取過的區域) 
extern QSemaphore g_qsemUsedBytes; // 控制環形緩沖區中的使用區(指生產者已填充數據,但消費者沒有讀取的區域) 
/**************************/ 

上述代碼中g_nDataSize、g_nBufferSize為全局常量,其他為全局變量。
復制代碼 代碼如下:

/**********res.cpp定義全局變量************/ 
#pragma once 
#include "res.h" 

// 定義全局變量 
char g_szBuffer[g_nBufferSize]; 
QSemaphore g_qsemFreeBytes(g_nBufferSize); 
QSemaphore g_qsemUsedBytes; 
/**************************/ 

在其他編譯單元中使用全局變量時只要包含其所在頭文件即可。
復制代碼 代碼如下:

/**********類ConsumerThread使用全局變量************/ 
#include "consumerthread.h" 
#include "res.h" 
#include <QDebug> 
ConsumerThread::ConsumerThread(QObject* parent) 
: QThread(parent) { 

ConsumerThread::ConsumerThread() { 


ConsumerThread::~ConsumerThread() { 

void ConsumerThread::run() { 
 for (int i = 0; i < g_nDataSize; i++) { 
  g_qsemUsedBytes.acquire();
  qDebug()<<"Consumer "<<g_szBuffer[i % g_nBufferSize]; 
  g_szBuffer[i % g_nBufferSize] = ' '; 
  g_qsemFreeBytes.release(); 
 } 
 qDebug()<<"&&Consumer Over"; 

/**************************/ 

也可以把全局變量的聲明和定義放在一起,這樣可以防止忘記了定義,如上面的extern char g_szBuffer[g_nBufferSize]; 然后把引用它的文件中的#include "res.h"換成extern char g_szBuffer[];。
但是這樣做很不好,因為你無法使用#include "res.h"(使用它,若達到兩次及以上,就出現重定義錯誤;注:即使在res.h中加#pragma once,或#ifndef也會出現重復定義,因為每個編譯單元是單獨的,都會對它各自進行定義),那么res.h聲明的其他函數或變量,你也就無法使用了,除非也都用extern修飾,這樣太麻煩,所以還是推薦使用.h中聲明,.cpp中定義的做法。
(5)靜態全局變量(static)
注意使用static修飾變量,就不能使用extern來修飾,即staticextern不可同時出現。
static修飾的全局變量的聲明與定義同時進行,即當你在頭文件中使用static聲明了全局變量,同時它也被定義了。
static修飾的全局變量的作用域只能是本身的編譯單元。在其他編譯單元使用它時,只是簡單的把其值復制給了其他編譯單元,其他編譯單元會另外開個內存保存它,在其他編譯單元對它的修改并不影響本身在定義時的值。即在其他編譯單元A使用它時,它所在的物理地址,和其他編譯單元B使用它時,它所在的物理地址不一樣,A和B對它所做的修改都不能傳遞給對方。
多個地方引用靜態全局變量所在的頭文件,不會出現重定義錯誤,因為在每個編譯單元都對它開辟了額外的空間進行存儲。
以下是Windows控制臺應用程序代碼示例:
復制代碼 代碼如下:

/***********res.h**********/ 
static char g_szBuffer[6] = "12345"; 
void fun(); 
/************************/ 

復制代碼 代碼如下:

/***********res.cpp**********/ 
#include "res.h" 
#include <iostream> 
using namespace std; 

void fun() { 
 for (int i = 0; i < 6; i++) { 
  g_szBuffer[i] = 'A' + i; 
 } 
 cout<<g_szBuffer<<endl; 

/************************/ 

復制代碼 代碼如下:

/***********test1.h**********/ 
void fun1(); 
/************************/ 

復制代碼 代碼如下:

/***********test1.cpp**********/ 
#include "test1.h" 
#include "res.h" 
#include <iostream> 
using namespace std; 

void fun1() { 
fun(); 

 for (int i = 0; i < 6; i++) { 
  g_szBuffer[i] = 'a' + i; 
 } 
 cout<<g_szBuffer<<endl; 

/************************/ 

復制代碼 代碼如下:

/***********test2.h**********/ 
void fun2(); 
/************************/ 

復制代碼 代碼如下:

/***********test2.cpp**********/ 
#include "test2.h" 
#include "res.h" 
#include <iostream> 
using namespace std; 

void fun2() { 
 cout<<g_szBuffer<<endl; 

/************************/ 

復制代碼 代碼如下:

/***********main.cpp**********/ 
#include "test1.h" 
#include "test2.h" 

int main() { 
 fun1(); 
 fun2(); 

 system("PAUSE"); 
 return 0; 

/************************/ 

運行結果如下:


按我們的直觀印象,認為fun1()和fun2()輸出的結果都為abcdef,可實際上fun2()輸出的確是初始值。然后我們再跟蹤調試,發現res、test1、test2中g_szBuffer的地址都不一樣,分別為0x0041a020、0x0041a084、0x0041a040,這就解釋了為什么不一樣。
注:一般定義static 全局變量時,都把它放在.cpp文件中而不是.h文件中,這樣就不會給其他編譯單元造成不必要的信息污染。
(6)全局常量(const)
const單獨使用時,其特性與static一樣(每個編譯單元中地址都不一樣,不過因為是常量,也不能修改,所以就沒有多大關系)。
const與extern一起使用時,其特性與extern一樣。
[code]
extern const char g_szBuffer[];      //寫入 .h中 
const char g_szBuffer[] = "123456"; // 寫入.cpp中 
[/code
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
少妇高潮久久77777| 亚洲bt天天射| 青青草成人在线| 精品二区三区线观看| 亚洲精品视频中文字幕| 久久天天躁夜夜躁狠狠躁2022| 亚洲性日韩精品一区二区| 亚洲天堂视频在线观看| 亚洲精品福利视频| 亚洲黄色av女优在线观看| 国产深夜精品福利| 国产精品一区二区三区久久久| 日韩一二三在线视频播| 97成人精品区在线播放| 国产精品人成电影| 亚洲欧美成人网| 日韩精品视频免费在线观看| 97在线观看免费高清| 深夜精品寂寞黄网站在线观看| 日韩av在线影视| 国产午夜精品久久久| 亚洲aaaaaa| 日韩av在线看| 欧美国产日韩二区| 日韩人在线观看| 亚洲va欧美va国产综合久久| 国产精品一二三视频| www.日本久久久久com.| 91干在线观看| 91精品国产91| 国产精品黄页免费高清在线观看| 亚洲曰本av电影| 亚洲国产精品一区二区久| 国产精品视频午夜| 久久九九精品99国产精品| 91国产视频在线播放| 日韩欧美国产黄色| 国产一区二区三区久久精品| 热久久99这里有精品| 亚洲人成电影网站色xx| 国产亚洲成精品久久| 日日骚av一区| 欧美国产精品日韩| 成人免费视频在线观看超级碰| 久久久久久免费精品| 色99之美女主播在线视频| 欧美日韩精品在线观看| 久久欧美在线电影| 成人av电影天堂| 日韩在线视频播放| 亚洲欧洲自拍偷拍| 在线a欧美视频| 亚洲精品视频免费在线观看| 欧美大荫蒂xxx| 日韩精品中文字幕久久臀| 亚洲在线视频福利| 久久久免费在线观看| 亚洲精品久久久久中文字幕欢迎你| 日韩美女毛茸茸| 国产精品永久免费观看| 91在线观看免费高清完整版在线观看| 日韩精品小视频| 欧美精品久久一区二区| 国产深夜精品福利| 国外视频精品毛片| 亚洲欧美日韩精品久久奇米色影视| 久久影视电视剧凤归四时歌| 97精品伊人久久久大香线蕉| 精品国产欧美一区二区五十路| 美女久久久久久久久久久| 久久久av网站| 九九久久综合网站| 欧美亚洲国产另类| 亚洲国产97在线精品一区| 96sao精品视频在线观看| 欧美一级黑人aaaaaaa做受| 国产69精品久久久久9| 欧美在线观看网址综合| 亚洲精品久久久久久久久| 国产精品久久久久久久久久久不卡| 亚洲国产一区自拍| 3344国产精品免费看| 最新国产成人av网站网址麻豆| 国产精品极品美女粉嫩高清在线| 日韩精品中文字幕在线播放| 亚洲视频在线观看视频| 国模gogo一区二区大胆私拍| 欧洲亚洲免费视频| 欧美午夜影院在线视频| 亚洲性无码av在线| 亚洲欧美日韩爽爽影院| 国产69久久精品成人看| 亚洲国产精品小视频| 国产伊人精品在线| 国产精品免费看久久久香蕉| 亚洲色图校园春色| 国产免费久久av| 国产精品福利片| 亚洲第一区在线观看| 久久精品国产精品| 一本久久综合亚洲鲁鲁| 国产亚洲视频中文字幕视频| 中文亚洲视频在线| 久久精品中文字幕电影| 成人精品aaaa网站| 国内外成人免费激情在线视频网站| 秋霞成人午夜鲁丝一区二区三区| 国产女同一区二区| 亚洲色图日韩av| 久久中文精品视频| 精品无码久久久久久国产| 日韩中文字幕在线播放| 中文欧美日本在线资源| 亚洲男人天堂古典| 日韩av一卡二卡| 欧美日韩中文在线观看| 岛国av午夜精品| 亚洲乱码av中文一区二区| 91精品中国老女人| 亚洲人成在线播放| 亚洲精品不卡在线| 欧美亚洲国产视频小说| 亚洲第一综合天堂另类专| 国产精品视频白浆免费视频| 日本精品一区二区三区在线播放视频| 欧美日韩国产丝袜美女| 美女性感视频久久久| 色噜噜狠狠狠综合曰曰曰| 国产精品黄页免费高清在线观看| 亚洲国产成人精品久久久国产成人一区| 亚洲美女福利视频网站| 日韩电影免费观看在线| 亚洲人成网7777777国产| 久久综合国产精品台湾中文娱乐网| 亚洲人成网站999久久久综合| 国语自产精品视频在线看抢先版图片| 最好看的2019年中文视频| 国产日韩精品在线| 国产丝袜一区二区| 久久精品视频在线播放| 91成人免费观看网站| 色综合久久久久久中文网| 欧美电影免费播放| 欧美激情一区二区三区成人| 午夜精品久久久久久久99热| 成年人精品视频| 欧美综合在线观看| 97成人精品区在线播放| 欧美一区二三区| 亚洲直播在线一区| 国产精品aaaa| 欧美亚洲在线视频| 久久久999精品视频| 91国在线精品国内播放| 亚洲欧洲国产精品| 亚州成人av在线| 成人福利视频在线观看| 亚洲成avwww人| 日韩有码在线视频| 韩国19禁主播vip福利视频| 日韩av中文在线| 欧美日韩国产中文精品字幕自在自线| 国产精品欧美一区二区| 国产欧亚日韩视频|