當(dāng)前位置:首頁 > 芯聞號 > 充電吧
[導(dǎo)讀]?extern "C"的主要作用就是為了能夠正確實(shí)現(xiàn)C++代碼調(diào)用其他C語言代碼。加上extern "C"后,會指示編譯器這部分代碼按C語言的進(jìn)行編譯,而不是C++的。由于C++支持函數(shù)重載,因此編譯

?extern "C"的主要作用就是為了能夠正確實(shí)現(xiàn)C++代碼調(diào)用其他C語言代碼。加上extern "C"后,會指示編譯器這部分代碼按C語言的進(jìn)行編譯,而不是C++的。由于C++支持函數(shù)重載,因此編譯器編譯函數(shù)的過程中會將函數(shù)的參數(shù)類型也加到編譯后的代碼中,而不僅僅是函數(shù)名;而C語言并不支持函數(shù)重載,因此編譯C語言代碼的函數(shù)時(shí)不會帶上函數(shù)的參數(shù)類型,一般之包括函數(shù)名。

???? 這個(gè)功能十分有用處,因?yàn)樵贑++出現(xiàn)以前,很多代碼都是C語言寫的,而且很底層的庫也是C語言寫的,為了更好的支持原來的C代碼和已經(jīng)寫好的C語言庫,需要在C++中盡可能的支持C,而extern "C"就是其中的一個(gè)策略。

這個(gè)功能主要用在下面的情況:

1、C++代碼調(diào)用C語言代碼

2、在C++的頭文件中使用

3、在多個(gè)人協(xié)同開發(fā)時(shí),可能有的人比較擅長C語言,而有的人擅長C++,這樣的情況下也會有用到

給出一個(gè)我設(shè)計(jì)的例子:

moduleA、moduleB兩個(gè)模塊,B調(diào)用A中的代碼,其中A是用C語言實(shí)現(xiàn)的,而B是利用C++實(shí)現(xiàn)的,下面給出一種實(shí)現(xiàn)方法:

//moduleA頭文件

#ifndef __MODULE_A_H //對于模塊A來說,這個(gè)宏是為了防止頭文件的重復(fù)引用

#define __MODULE_A_H

int fun(int, int);

#endif

//moduleA實(shí)現(xiàn)文件moduleA.C //模塊A的實(shí)現(xiàn)部分并沒有改變

#include"moduleA"

int fun(int a, int b)

{

return a+b;

}

//moduleB頭文件

#idndef __MODULE_B_H //很明顯這一部分也是為了防止重復(fù)引用

#define __MODULE_B_H

#ifdef __cplusplus //而這一部分就是告訴編譯器,如果定義了__cplusplus(即如果是cpp文件, extern "C"{ //因?yàn)閏pp文件默認(rèn)定義了該宏),則采用C語言方式進(jìn)行編譯

#include"moduleA.h"

#endif

… //其他代碼

#ifdef __cplusplus

}

#endif

#endif

//moduleB實(shí)現(xiàn)文件 moduleB.cpp //B模塊的實(shí)現(xiàn)也沒有改變,只是頭文件的設(shè)計(jì)變化了

#include"moduleB.h"

int main()

{

cout<<fun(2,3)<<endl;

}

下面是詳細(xì)的介紹:

由于C、C++編譯器對函數(shù)的編譯處理是不完全相同的,尤其對于C++來說,支持函數(shù)的重載,編譯后的函數(shù)一般是以函數(shù)名和形參類型來命名的。

例如函數(shù)void fun(int, int),編譯后的可能是(不同編譯器結(jié)果不同)_fun_int_int(不同編譯器可能不同,但都采用了類似的機(jī)制,用函數(shù)名和參數(shù)類型來命名編譯后的函數(shù)名);而C語言沒有類似的重載機(jī)制,一般是利用函數(shù)名來指明編譯后的函數(shù)名的,對應(yīng)上面的函數(shù)可能會是_fun這樣的名字。

看下面的一個(gè)面試題:為什么標(biāo)準(zhǔn)頭文件都有類似的結(jié)構(gòu)?

#ifndef __INCvxWorksh /*防止該頭文件被重復(fù)引用*/

#define __INCvxWorksh

#ifdef __cplusplus???????????? //告訴編譯器,這部分代碼按C語言的格式進(jìn)行編譯,而不是C++的

extern "C"{

#endif

/*…*/

#ifdef __cplusplus

}

#endif

#endif /*end of __INCvxWorksh*/

分析:

顯然,頭文件中編譯宏"#ifndef __INCvxWorksh 、#define __INCvxWorksh、#endif"(即上面代碼中的藍(lán)色部分)的作用是為了防止該頭文件被重復(fù)引用那么

#ifdef __cplusplus (其中__cplusplus是cpp中自定義的一個(gè)宏?。?!)

extern "C"{

#endif

#ifdef __cplusplus

}

#endif

的作用是什么呢?

extern "C"包含雙重含義,從字面上可以知道,首先,被它修飾的目標(biāo)是"extern"的;其次,被它修飾的目標(biāo)代碼是"C"的。

被extern "C"限定的函數(shù)或變量是extern類型的

extern是C/C++語言中表明函數(shù)和全局變量的作用范圍的關(guān)鍵字,該關(guān)鍵字告訴編譯器,其申明的函數(shù)和變量可以在本模塊或其他模塊中使用。

記住,下面的語句:

extern int a; 僅僅是一個(gè)變量的聲明,其并不是在定義變量a,并未為a分配空間。變量a在所有模塊中作為一種全局變量只能被定義一次,否則會出錯(cuò)。

通常來說,在模塊的頭文件中對本模塊提供給其他模塊引用的函數(shù)和全局變量以關(guān)鍵字extern生命。例如,如果模塊B要引用模塊A中定義的全局變量和函數(shù)時(shí)只需包含模塊A的頭文件即可。這樣模塊B中調(diào)用模塊A中的函數(shù)時(shí),在編譯階段,模塊B雖然找不到該函數(shù),但并不會報(bào)錯(cuò);它會在鏈接階段從模塊A編譯生成的目標(biāo)代碼中找到該函數(shù)。

extern對應(yīng)的關(guān)鍵字是static,static表明變量或者函數(shù)只能在本模塊中使用,因此,被static修飾的變量或者函數(shù)不可能被extern C修飾。

被extern "C"修飾的變量和函數(shù)是按照C語言方式進(jìn)行編譯和鏈接的:這點(diǎn)很重要?。。?!

上面也提到過,由于C++支持函數(shù)重載,而C語言不支持,因此函數(shù)被C++編譯后在符號庫中的名字是與C語言不同的;C++編譯后的函數(shù)需要加上參數(shù)的類型才能唯一標(biāo)定重載后的函數(shù),而加上extern "C"后,是為了向編譯器指明這段代碼按照C語言的方式進(jìn)行編譯

未加extern "C"聲明時(shí)的鏈接方式:

//模塊A頭文件 moduleA.h

#idndef _MODULE_A_H

#define _MODULE_A_H

int foo(int x, int y);

#endif

在模塊B中調(diào)用該函數(shù):

//模塊B實(shí)現(xiàn)文件 moduleB.cpp

#include"moduleA.h"

foo(2,3);

實(shí)際上,在鏈接階段,連接器會從模塊A生成的目標(biāo)文件moduleA.obj中找_foo_int_int這樣的符號?。?!,顯然這是不可能找到的,因?yàn)閒oo()函數(shù)被編譯成了_foo的符號,因此會出現(xiàn)鏈接錯(cuò)誤。

常見的做法可以參考下面的一個(gè)實(shí)現(xiàn):

moduleA、moduleB兩個(gè)模塊,B調(diào)用A中的代碼,其中A是用C語言實(shí)現(xiàn)的,而B是利用C++實(shí)現(xiàn)的,下面給出一種實(shí)現(xiàn)方法:

//moduleA頭文件

#ifndef __MODULE_A_H //對于模塊A來說,這個(gè)宏是為了防止頭文件的重復(fù)引用

#define __MODULE_A_H

int fun(int, int);

#endif

//moduleA實(shí)現(xiàn)文件moduleA.C //模塊A的實(shí)現(xiàn)部分并沒有改變

#include"moduleA"

int fun(int a, int b)

{

return a+b;

}

//moduleB頭文件

#idndef __MODULE_B_H //很明顯這一部分也是為了防止重復(fù)引用

#define __MODULE_B_H

#ifdef __cplusplus //而這一部分就是告訴編譯器,如果定義了__cplusplus(即如果是cpp文件, extern "C"{ //因?yàn)閏pp文件默認(rèn)定義了該宏),則采用C語言方式進(jìn)行編譯

#include"moduleA.h"

#endif

… //其他代碼

#ifdef __cplusplus

}

#endif

#endif

//moduleB實(shí)現(xiàn)文件 moduleB.cpp //B模塊的實(shí)現(xiàn)也沒有改變,只是頭文件的設(shè)計(jì)變化了

#include"moduleB.h"

int main()

{

cout<<fun(2,3)<<endl;

}

extern "C"的使用要點(diǎn)

1. 可以是單一語句

??? extern "C" double sqrt(double);

2. 可以是復(fù)合語句, 相當(dāng)于復(fù)合語句中的聲明都加了extern "C"

??? extern "C"

?? {

????? double sqrt(double);

????? int min(int, int);

? }

3.可以包含頭文件,相當(dāng)于頭文件中的聲明都加了extern "C"

?? extern "C"

? {

??? #i nclude

? }

4. 不可以將extern "C" 添加在函數(shù)內(nèi)部

5. 如果函數(shù)有多個(gè)聲明,可以都加extern "C", 也可以只出現(xiàn)在第一次聲明中,后面的聲明會接受第一個(gè)鏈接指示符的規(guī)則。

6. 除extern "C", 還有extern "FORTRAN" 等。

本站聲明: 本文章由作者或相關(guān)機(jī)構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點(diǎn),本站亦不保證或承諾內(nèi)容真實(shí)性等。需要轉(zhuǎn)載請聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請及時(shí)聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車的華為或?qū)⒋呱龈蟮莫?dú)角獸公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關(guān)鍵字: 阿維塔 塞力斯 華為

加利福尼亞州圣克拉拉縣2024年8月30日 /美通社/ -- 數(shù)字化轉(zhuǎn)型技術(shù)解決方案公司Trianz今天宣布,該公司與Amazon Web Services (AWS)簽訂了...

關(guān)鍵字: AWS AN BSP 數(shù)字化

倫敦2024年8月29日 /美通社/ -- 英國汽車技術(shù)公司SODA.Auto推出其旗艦產(chǎn)品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認(rèn)證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開發(fā)耗時(shí)1.5...

關(guān)鍵字: 汽車 人工智能 智能驅(qū)動 BSP

北京2024年8月28日 /美通社/ -- 越來越多用戶希望企業(yè)業(yè)務(wù)能7×24不間斷運(yùn)行,同時(shí)企業(yè)卻面臨越來越多業(yè)務(wù)中斷的風(fēng)險(xiǎn),如企業(yè)系統(tǒng)復(fù)雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務(wù)連續(xù)性,提升韌性,成...

關(guān)鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報(bào)道,騰訊和網(wǎng)易近期正在縮減他們對日本游戲市場的投資。

關(guān)鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會開幕式在貴陽舉行,華為董事、質(zhì)量流程IT總裁陶景文發(fā)表了演講。

關(guān)鍵字: 華為 12nm EDA 半導(dǎo)體

8月28日消息,在2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會上,華為常務(wù)董事、華為云CEO張平安發(fā)表演講稱,數(shù)字世界的話語權(quán)最終是由生態(tài)的繁榮決定的。

關(guān)鍵字: 華為 12nm 手機(jī) 衛(wèi)星通信

要點(diǎn): 有效應(yīng)對環(huán)境變化,經(jīng)營業(yè)績穩(wěn)中有升 落實(shí)提質(zhì)增效舉措,毛利潤率延續(xù)升勢 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務(wù)引領(lǐng)增長 以科技創(chuàng)新為引領(lǐng),提升企業(yè)核心競爭力 堅(jiān)持高質(zhì)量發(fā)展策略,塑強(qiáng)核心競爭優(yōu)勢...

關(guān)鍵字: 通信 BSP 電信運(yùn)營商 數(shù)字經(jīng)濟(jì)

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺與中國電影電視技術(shù)學(xué)會聯(lián)合牽頭組建的NVI技術(shù)創(chuàng)新聯(lián)盟在BIRTV2024超高清全產(chǎn)業(yè)鏈發(fā)展研討會上宣布正式成立。 活動現(xiàn)場 NVI技術(shù)創(chuàng)新聯(lián)...

關(guān)鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會上,軟通動力信息技術(shù)(集團(tuán))股份有限公司(以下簡稱"軟通動力")與長三角投資(上海)有限...

關(guān)鍵字: BSP 信息技術(shù)
關(guān)閉
關(guān)閉