C++中的extern聲明變量詳解
extern聲明變量無外乎如下兩種:
1、聲明全局變量
2、聲明函數(shù)
今天我們只談extern,什么const、static之類等等與之相關(guān)或不相關(guān)的一律忽略,下面就分別對以上兩種情況一一講解
聲明和定義
既然提到extern聲明變量,那我們就必須搞清楚聲明和定義的區(qū)別。
這里我們將普通數(shù)據(jù)變量和函數(shù)統(tǒng)稱變量。從內(nèi)存分配角度來說,聲明和定義的區(qū)別在于聲明一個變量不會分配內(nèi)存,而定義一個變量會分配內(nèi)存。一個變量可以被聲明多次,但是只能被定義一次。
基于以上前提,我們可以把聲明和定義類比為指針和內(nèi)存的關(guān)系。我們知道,指針其實就是指向內(nèi)存的一個符號,變量的定義就好比一塊內(nèi)存區(qū)域,而聲明就好比它的指針,可以有多個指針指向同一個內(nèi)存區(qū)域,而一個指針只能指向一個內(nèi)存區(qū)域,這樣就很好理解為什么變量只能被定義一次,如果被定義多次,那就會分配多個內(nèi)存,這樣你通過變量的聲明到底去找哪塊內(nèi)存區(qū)域呢,這會是個問題。
對于數(shù)據(jù)來說,聲明和定義往往是同時存在的,比如下面的一行語句
int data;
這樣既聲明了data同時也定義了data,怎樣做到只聲明而不定義呢,用extern就可以了
extern int data;
對于函數(shù)來說,聲明和定義就很容易區(qū)分了,一般我們會將聲明放在頭文件而將定義放在源文件里
void hello();
這是一個函數(shù)的聲明,而
void hello()
{
printf("hello world!\n");
}
這是一個函數(shù)的定義。當(dāng)然,函數(shù)的聲明和定義也可以同時發(fā)生,如果我們沒有頭文件而只有源文件,并且在源文件里并沒有void hello();這樣的語句,那么這個函數(shù)的聲明和定義就同時發(fā)生了,此時如果我們在原文件里想要調(diào)用函數(shù)hello(),你調(diào)用的代碼必須在函數(shù)定義之后。
其實上面的要點只在于一句話:使用變量之前必須聲明,聲明可以有多次,而定義只能有一次。記住這句話,后面的就都很容易理解了。
extern聲明全局變量
我們先來看如下例子,現(xiàn)有三個文件:test.h, test.cpp, main.cpp,其中main.cpp和test.cpp需要共享一個變量g_name,三個文件的內(nèi)容如下
/* test.h */
#ifndef _TEST_H_
#define _TEST_H_
#include <string>
std::string g_name;
void hello();
#endif
/* test.cpp */
#include <stdio.h>
#include "test.h"
void hello()
{
printf("hello %s!\n", g_name.c_str());
}
/* main.cpp */
#include "test.h"
std::string g_name;
int main()
{
g_name = "Handy";
hello();
return 0;
}
三者關(guān)系為,test.cpp包含了test.h,main.cpp也包含了test.h,這里的包含其實就是include。我們執(zhí)行編譯命令
g++ main.cpp test.cpp
編譯報錯redefinition of 'g_name',說的是g_name被重定義了
我們看一下g_name出現(xiàn)的地方,一個是在test.h里,一個是在main.cpp里,兩條語句都是std::string g_name,前面我們已經(jīng)說過,這樣的方式既聲明也定義了變量,那g_name是如何被重定義的呢,首先我們需要理解include的含義,我們可以將include一個頭文件理解為在該行展開頭文件里的所有代碼,由于main.cpp包含了test.h,我們在那一行將test.h的內(nèi)容展開,就會發(fā)現(xiàn)main.cpp里有兩句std::string g_name;所以在main.cpp里,g_name被定義了兩次。
由于我們可以將include頭文件理解為展開代碼,所以編譯的時候其實不需要指定頭文件,只需要源文件就夠了。需要注意的是,重定義并不是指在同一個原文件里定義多次,而是指在整個代碼空間里,比如上面的例子是就是指在test.cpp和main.cpp里,其實上面的例子里g_name是被重定義了三次,其中test.cpp里一次,main.cpp里兩次。
那上面重定義的問題怎么解決呢,很簡答,將test.h里的std::string g_name;改為extern std::string g_name;就可以了,由于extern語句只聲明變量而不定義變量,因此test.cpp和main.cpp展開頭文件后,也只是將g_name聲明了兩次,而真正的定義還是在main.cpp里
extern聲明函數(shù)
還是上面的例子,我們怎么在main.cpp里不包含頭文件就可以調(diào)用hello函數(shù)呢,既然今天的主題是extern,不用提醒也知道,使用extern就可以了,代碼如下
/* test.cpp */
#include <string>
#include <stdio.h>
// 聲明g_name
extern std::string g_name;
// 聲明和定義void hello()
void hello()
{
printf("hello %s!\n", g_name.c_str());
}
/* main.cpp */
#include <string>
// 聲明和定義g_name
std::string g_name;
// 聲明void hello()
extern void hello();
int main()
{
g_name = "Handy"
hello();
return 0;
}
注意這里用到extern聲明變量和函數(shù)兩種場景,我分別在語句后面做了注釋。編譯命令如下
g++ main.cpp test.cpp
這里我們并沒有用到頭文件,但是依然可以在不同文件間共享變量和函數(shù),這一切都是extern的功勞!
總結(jié)
要了解extern主要搞清以下幾個概念:
1、聲明和定義的區(qū)別。全局代碼空間里,變量可以有多個聲明,但只能有一個定義
2、include頭文件等同于展開頭文件里的代碼
了解了以上兩點,再來分析extern的用法,是不是就會清晰很多了
您可能感興趣的文章
- 04-02c語言沒有round函數(shù) round c語言
- 01-10深入理解C++中常見的關(guān)鍵字含義
- 01-10使用C++實現(xiàn)全排列算法的方法詳解
- 01-10深入Main函數(shù)中的參數(shù)argc,argv的使用詳解
- 01-10c++中inline的用法分析
- 01-10如何尋找數(shù)組中的第二大數(shù)
- 01-10用C++實現(xiàn)DBSCAN聚類算法
- 01-10全排列算法的非遞歸實現(xiàn)與遞歸實現(xiàn)的方法(C++)
- 01-10C++大數(shù)模板(推薦)
- 01-10淺談C/C++中的static與extern關(guān)鍵字的使用詳解


閱讀排行
本欄相關(guān)
- 04-02c語言函數(shù)調(diào)用后清空內(nèi)存 c語言調(diào)用
- 04-02func函數(shù)+在C語言 func函數(shù)在c語言中
- 04-02c語言的正則匹配函數(shù) c語言正則表達
- 04-02c語言用函數(shù)寫分段 用c語言表示分段
- 04-02c語言中對數(shù)函數(shù)的表達式 c語言中對
- 04-02c語言編寫函數(shù)冒泡排序 c語言冒泡排
- 04-02c語言沒有round函數(shù) round c語言
- 04-02c語言分段函數(shù)怎么求 用c語言求分段
- 04-02C語言中怎么打出三角函數(shù) c語言中怎
- 04-02c語言調(diào)用函數(shù)求fibo C語言調(diào)用函數(shù)求
隨機閱讀
- 01-10C#中split用法實例總結(jié)
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
- 01-10使用C語言求解撲克牌的順子及n個骰子
- 01-11ajax實現(xiàn)頁面的局部加載
- 01-10SublimeText編譯C開發(fā)環(huán)境設(shè)置
- 01-10delphi制作wav文件的方法
- 08-05織夢dedecms什么時候用欄目交叉功能?
- 04-02jquery與jsp,用jquery
- 08-05dedecms(織夢)副欄目數(shù)量限制代碼修改
- 08-05DEDE織夢data目錄下的sessions文件夾有什