成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

C++中是如何調(diào)用C接口的?

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

[[335900]]

前言

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

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

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

首先提供一個C接口:

 

  1. //來源:公眾號【編程珠璣】 
  2. // 博客:https://www.yanbinghu.com 
  3. //test.c 
  4. #include"test.h" 
  5. void testCfun() 
  6.     printf("I am c fun\n"); 
  7.     return

為了簡化,我們在這里就不將它做成靜態(tài)庫或者動態(tài)庫了,有興趣的可以參考《靜態(tài)庫制作》自行嘗試。我們在這里編譯成C目標(biāo)文件:

 

  1. gcc -c test.c 

另外提供一個頭文件test.h:

 

  1. #include<stdio.h> 
  2. void testCfun(); 

我們的C++代碼調(diào)用如下:

 

  1. //來源:公眾號【編程珠璣】 博客:https://www.yanbinghu.com 
  2. //main.cpp 
  3. #include"test.h" 
  4. #include<iostream> 
  5. using namespace std; 
  6. int main(void) 
  7.     /*調(diào)用C接口*/ 
  8.     cout<<"start to call c function"<<endl; 
  9.     testCfun(); 
  10.     cout<<"end to call c function"<<endl; 
  11.     return 0; 

編譯:

 

  1. $ g++ -o main main.cpp test.o 
  2. /tmp/ccmwVJqM.o: In function `main': 
  3. main.cpp:(.text+0x21): undefined reference to `testCfun()' 
  4. collect2: error: ld returned 1 exit status 

很不幸,最后的鏈接報(bào)錯了,說找不到testCfun,但是我們確實(shí)定義了這個函數(shù)。為什么會找不到呢?現(xiàn)在你還會認(rèn)為C++直接就可以調(diào)用C接口了嗎?

真相

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

 

  1. $ nm test.o|grep testCfun 
  2. 0000000000000000 T testCfun 
  3. $ nm main.o|grep testCfun 
  4.                 U _Z8testCfunv 

所以它們兩個能鏈接在一起才真是奇怪了呢!名字都不同,還怎么鏈接?

如何處理

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

 

  1. #include<stdio.h> 
  2. extern "C"
  3. void testCfun(); 

這里用extern "C"將testCfun接口包裹起來,告訴編譯器,這里的是C代碼哈,你要按C代碼的方式處理。再次編譯:

 

  1. $ g++ -o main main.cpp test.o 
  2. $ ./main 
  3. start to call c function 
  4. I am c fun 
  5. end to call c function 

看終端輸出,完美!

優(yōu)化

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

 

  1. //main.c 
  2. #include"test.h" 
  3. int main(void) 
  4.     /*調(diào)用C接口*/ 
  5.     testCfun(); 
  6.     return 0; 

編譯:

 

  1. $ gcc -o main main.c test.c 
  2. In file included from main.c:2:0: 
  3. test.h:2:8: error: expected identifier or '(' before string constant 
  4.  extern "C"
  5.         ^ 
  6. In file included from test.c:2:0: 
  7. test.h:2:8: error: expected identifier or '(' before string constant 
  8.  extern "C"

不出意外,又報(bào)錯了,很顯然,C語言中并沒有extern "C"這樣的寫法,所以為了能使得test.c的代碼既能被C++調(diào)用,也能被C調(diào)用,需要改寫成下面這樣:

 

  1. #include<stdio.h> 
  2. #ifdef __cplusplus 
  3. extern "C"
  4. #endif 
  5.  
  6. void testCfun(); 
  7.  
  8. #ifdef __cplusplus 
  9. #endif 

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

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

問題

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

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

 

  1. $ g++ -E main.i main.cpp 

去生成的main.i文件中找一找,是不是有extern "C"。

總結(jié)

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

那么問題來了,C又該如何調(diào)用C++的接口呢?

責(zé)任編輯:龐桂玉 來源: C語言與C++編程
相關(guān)推薦

2019-08-28 14:21:39

C++C接口代碼

2021-10-11 11:53:07

C++接口代碼

2010-01-28 13:35:41

調(diào)用C++函數(shù)

2010-01-20 09:54:27

C++數(shù)據(jù)類型

2025-05-20 10:00:00

C++命名空間別名代碼

2010-01-21 14:07:14

CC++聲明

2010-01-21 09:34:57

C++語法

2010-01-27 16:05:06

C++堆棧

2010-01-26 15:51:06

C++變量

2010-02-01 13:25:32

Python腳本

2014-09-19 10:46:36

LuaCC++

2023-11-09 23:31:02

C++函數(shù)調(diào)用

2011-05-18 17:15:45

2010-01-20 10:19:55

C++數(shù)組

2010-01-25 10:25:19

C++變量

2010-01-27 17:16:52

C++構(gòu)造函數(shù)

2010-01-28 16:31:54

C++類型

2010-01-21 13:33:44

C++基類

2020-08-21 13:20:36

C++If ElseLinux

2010-01-26 10:42:26

C++函數(shù)
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號

主站蜘蛛池模板: 二区不卡 | 久久久久一区二区 | 三级av网址 | 亚洲精品久久久久久久久久久久久 | 欧洲av一区 | 国产一区亚洲 | 欧美激情精品久久久久久变态 | 免费啪啪| a级黄色片在线观看 | 久草福利| 成人精品系列 | 精品99爱视频在线观看 | 欧美成人a | 99热这里有精品 | 久久久久久国产 | 日韩欧美在线免费观看 | 欧美在线| 日本一区二区视频 | 久久精品国产亚洲一区二区三区 | 91精品久久久久久久久久入口 | 久久久久中文字幕 | 亚洲 中文 欧美 日韩 在线观看 | 999久久久久久久久6666 | 欧美三级电影在线播放 | 国产精品久久久亚洲 | av网站免费在线观看 | 九九导航 | 成人伊人 | 国产一区二区三区欧美 | 国产偷久久一级精品60部 | 日韩在线观看中文字幕 | 久久国| 亚洲免费在线 | 国产精品久久久久久久久免费相片 | 久久国产精品视频 | 亚洲欧美日本在线 | av中文字幕在线 | 91原创视频 | 最新中文字幕久久 | 国产视频欧美 | 一区二区不卡视频 |