當(dāng)前位置:首頁(yè) > 公眾號(hào)精選 > C語(yǔ)言與CPP編程
[導(dǎo)讀]來(lái)源:公眾號(hào)【編程珠璣】作者:守望先生前言如何在C代碼中調(diào)用寫(xiě)好的C接口?你可能會(huì)奇怪,C不是兼容C嗎?直接調(diào)用不就可以了?這里我們先按下不表,先看看C如何調(diào)用C代碼接口。C如何調(diào)用C接口為什么會(huì)有這樣的情況呢?想象一下,有些接口是用C實(shí)現(xiàn)的,并提供了庫(kù),那么C中該如何使用呢?我...

來(lái)源:公眾號(hào)【編程珠璣】

作者:守望先生

前言

如何在C 代碼中調(diào)用寫(xiě)好的C接口?你可能會(huì)奇怪,C 不是兼容C嗎?直接調(diào)用不就可以了?這里我們先按下不表,先看看C 如何調(diào)用C代碼接口。

C 如何調(diào)用C接口

為什么會(huì)有這樣的情況呢?想象一下,有些接口是用C實(shí)現(xiàn)的,并提供了庫(kù),那么C 中該如何使用呢?我們先不做任何區(qū)別對(duì)待,看看普通情況下會(huì)發(fā)生什么意想不到的事情。
首先提供一個(gè)C接口:

//來(lái)源:公眾號(hào)【編程珠璣】//?博客:https://www.yanbinghu.com
//test.c
#include"test.h"
void?testCfun()
{
????printf("I?am?c?fun\n");
????return;
}
為了簡(jiǎn)化,我們?cè)谶@里就不將它做成靜態(tài)庫(kù)或者動(dòng)態(tài)庫(kù)了,有興趣的可以參考《靜態(tài)庫(kù)制作》自行嘗試。我們?cè)谶@里編譯成C目標(biāo)文件:

gcc?-c?test.c
另外提供一個(gè)頭文件test.h:

#include
void?testCfun();
我們的C 代碼調(diào)用如下:

//來(lái)源:公眾號(hào)【編程珠璣】?博客:https://www.yanbinghu.com
//main.cpp
#include"test.h"
#include
using?namespace?std;
int?main(void)
{
????/*調(diào)用C接口*/
????cout<<"start?to?call?c?function"<<endl;
????testCfun();
????cout<<"end?to?call?c?function"<<endl;
????return?0;
}
編譯:

$?g ?-o?main?main.cpp?test.o
/tmp/ccmwVJqM.o:?In?function?`main':
main.cpp:(.text 0x21):?undefined?reference?to?`testCfun()'
collect2:?error:?ld?returned?1?exit?status
很不幸,最后的鏈接報(bào)錯(cuò)了,說(shuō)找不到testCfun,但是我們確實(shí)定義了這個(gè)函數(shù)。為什么會(huì)找不到呢?現(xiàn)在你還會(huì)認(rèn)為C 直接就可以調(diào)用C接口了嗎?

真相

我們都知道,C 中函數(shù)支持重載,而C并不支持。C 為了支持函數(shù)重載,它在“生成”函數(shù)符號(hào)信息時(shí),不能僅僅通過(guò)函數(shù)名,因?yàn)橹剌d函數(shù)的函數(shù)名都是一樣的,所以它還要根據(jù)入?yún)ⅲ臻g等信息來(lái)確定唯一的函數(shù)簽名?;蛘哒f(shuō)C 生成函數(shù)簽名的方式與C不一致,所以即便是函數(shù)名一樣,對(duì)于C和C 來(lái)說(shuō),它們最終的函數(shù)簽名還是不一樣。當(dāng)然這里又是另外一回事了,我們不細(xì)說(shuō)。我們看看兩個(gè)文件里的函數(shù)符號(hào)有什么區(qū)別:

$?nm?test.o|grep?testCfun
0000000000000000?T?testCfun
$?nm?main.o|grep?testCfun
????????????????U?_Z8testCfunv
所以它們兩個(gè)能鏈接在一起才真是奇怪了呢!名字都不同,還怎么鏈接?

如何處理

那么如何處理呢?很顯然,我們必須告訴鏈接器,這是一個(gè)C接口,而不是C 接口,所以需要加入 extern C,我們修改test.h

#include
extern?"C"{
void?testCfun();
}
這里用extern "C"將testCfun接口包裹起來(lái),告訴編譯器,這里的是C代碼哈,你要按C代碼的方式處理。再次編譯:

$?g ?-o?main?main.cpp?test.o
$?./main
start?to?call?c?function
I?am?c?fun
end?to?call?c?function
看終端輸出,完美!

優(yōu)化

雖然上面的C接口可以被C 正常調(diào)用了,但是如果這個(gè)C接口要被C代碼調(diào)用呢?增加main.c內(nèi)容如下

//main.c
#include"test.h"
int?main(void)
{
????/*調(diào)用C接口*/
????testCfun();
????return?0;
}
編譯:

$?gcc?-o?main?main.c?test.c
In?file?included?from?main.c:2:0:
test.h:2:8:?error:?expected?identifier?or?'('?before?string?constant
?extern?"C"{
????????^
In?file?included?from?test.c:2:0:
test.h:2:8:?error:?expected?identifier?or?'('?before?string?constant
?extern?"C"{
不出意外,又報(bào)錯(cuò)了,很顯然,C語(yǔ)言中并沒(méi)有extern "C"這樣的寫(xiě)法,所以為了能使得test.c的代碼既能被C 調(diào)用,也能被C調(diào)用,需要改寫(xiě)成下面這樣:

#include
#ifdef?__cplusplus
extern?"C"{
#endif

void?testCfun();

#ifdef?__cplusplus
}
#endif
這里通過(guò)__cplusplus宏來(lái)控制是否需要extern “C”,如果是C 編譯器,那么extern "C"部分就會(huì)被預(yù)處理進(jìn)去,這樣test.c代碼就可以既用于C ,也可以用于C啦。

趕快去你的C項(xiàng)目代碼頭文件中看看,是不是也有這樣的代碼段呢?

來(lái)源:公眾號(hào)【編程珠璣】,專(zhuān)注但不限于分享計(jì)算機(jī)編程基礎(chǔ),Linux,C語(yǔ)言,C ,數(shù)據(jù)結(jié)構(gòu)與算法,工具,資源等編程相關(guān)[原創(chuàng)]技術(shù)文章。博客:https://www.yanbinghu.com

問(wèn)題

為什么我們?cè)贑 代碼中可以直接調(diào)用一些標(biāo)準(zhǔn)C庫(kù)函數(shù)呢?即使你在main函數(shù)中調(diào)用printf等函數(shù),它也不會(huì)出現(xiàn)鏈接錯(cuò)誤。因?yàn)閹?kù)函數(shù)已經(jīng)有了類(lèi)似的處理了。

如果你還是不確定,你可以先預(yù)處理:

$?g ?-E?main.i?main.cpp
去生成的main.i文件中找一找,是不是有extern "C"。

總結(jié)

C 支持重載,而C不支持,C 并不能直接調(diào)用C代碼寫(xiě)好的接口,因此如果你的C代碼想要能夠被C調(diào)用,也想被C 調(diào)用,那么別忘了extern "C"。

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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