您好,登錄后才能下訂單哦!
前言
本文主要給大家介紹了關于C/C++混合編程extern “C”使用的相關內容,分享出來供大家參考學習,下面話不多說了,來一起看看詳細的介紹吧。
首先要明白:
C++號稱是C語言的超集,也確實,從語言的基本語法上,C++是包含所有C語言的語法的,而且C++為了兼容C,連C語言的標準庫也被納入到C++的標準庫中,比如在C++中我們仍然可以使用<stdio.h>
,它就是C++標準庫的一部分(注意最好用新的標準<cstdio>
,而不是老的<stdio>
)。
但是,C++和C語言的編譯器在有些地方還是有差別的。比如,C++是支持面向對象的特性(盡管C++被稱為不徹底的面向對象語言),面向對象就要支持函數重載,而函數重載的實現和C++編譯器是分不開的。
比如有:
int max(int a, int b);
和
double max(double a,double b);
這兩個函數。
這兩個函數同時存在,在C語言中是不可以的,因為C語言不支持重載,但是在C++中完全沒有問題,因為C++是支持重載的。
那么為什么會這樣呢?
這要從編譯器的角度來看:
C編譯器將int max(int a,int b)
函數編譯后,在符號庫中的名字為_max(不同編譯器可能有不同的形式)
C編譯器將double max(double a,double b)
函數編譯后,在符號庫中的名字也為_max。所以在C語言中,這兩個函數是不能共存的的,編譯時會提示錯誤。
但是C++編譯器則不同
C++編譯器將int max(int a, int b)
編譯后,在符號表中的名字為:
_max_int_int
而另一個則為_max_double_double
。這兩者并不沖突。
這也是C++支持函數重載的根本原因。不過,這也從另一面說明了:函數重載其實不是多態,頂多算是編譯時多態。
在C++和C混合編程時:
在C++和C語言混合編程時,前面已經說過,C++的語法是完全包含C語言的語法的,所以不用擔心語法上出現什么問題。出現問題的主要原因在編譯和鏈接時。
思考這樣一件事情:
我們在Linux下編程的時候,經常會用到內核的庫,我們知道Linux內核使用C語言寫的。如果這些庫采用了C語言的編譯器進行編譯,而我們的程序是用C++開發的,要用C++進行編譯,那么會出現什么問題?
比如:
我們有個.c文件p.c,內容如下:
#include <stdio.h> void print(int a,int b) { printf("這里調用的是C語言的函數:%d,%d\n",a,b); }
我們用gcc –c
命令編譯它,生成p.o文件
然后我們又p.h文件,里面是print函數的原型聲明,如下:
void print(int a,int b);
我們又有main.cpp文件,內容如下
#include <iostream> using namespace std; #include "p.h" int main() { cout<<"現在調用C語言函數\n"; print(3,4); return 0; }
這次我們用g++ -c進行編譯,生成main.o文件
然后我們用g++鏈接兩個.o文件,這時我們會發現,報錯:
這時為什么呢?
因為:
p.c我們使用的是C語言的編譯器gcc進行編譯的,其中的函數print編譯之后,在符號表中的名字為 _print
而我們鏈接的時候采用的是g++進行鏈接,也就是C++鏈接方式,程序在運行到調用print函數的代碼時,會在符號表中尋找_print_int_int(是按照C++的鏈接方法來尋找的,所以是找_print_int_int而不是找_print)的名字,發現找不到,所以會t提示“未定義的引用”
此時如果我們在對print的聲明中加入 extern “C” ,這個時候,g++編譯器就會按照C語言的鏈接方式進行尋找,也就是在符號表中尋找_print,這個時候是可以找到的,是不會報錯的。比如我們修改p.h為:
extern "C" void print(int a,int b);
這是結果為:
所以extern “C” 用到函數聲明之前,它的作用就是告訴編譯器,對于該函數的鏈接要采用C語言編譯器的鏈接方式,也就是告訴編譯器找_fun,而不是_fun_int_int。
如果有多個函數聲明都需要在前面加extern “C”,那可以用extern “C”{}的形式。
總結
以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對億速云的支持。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。