深入理解C/C++混合編程
一、extern“C”的作用(最重點(diǎn))
1. extern "C"的真實(shí)目的是實(shí)現(xiàn)類(lèi)C和C++的混合編程。extern “C”是由C++提供的一個(gè)連接交換指定符號(hào),用于告訴C++這段代碼是C函數(shù)。extern “C”后面的函數(shù)不使用的C++的名字修飾,而是用C。這是因?yàn)?/SPAN>C++編譯后庫(kù)中函數(shù)名會(huì)變得很長(zhǎng),與C生成的不一致,造成C++不能直接調(diào)用C函數(shù)。
2.C++語(yǔ)言支持函數(shù)重載,C語(yǔ)言不支持函數(shù)重載。函數(shù)被C++編譯后在庫(kù)中的名字與C語(yǔ)言的不同。假設(shè)某個(gè)函數(shù)的原型為:void foo(int x, int y);該函數(shù)被C編譯器編譯后在庫(kù)中的名字為_foo,而C++編譯器則會(huì)產(chǎn)生像_foo_int_int之類(lèi)的名字。C++提供了C連接交換指定符號(hào)extern“C”來(lái)解決名字匹配問(wèn)題。
3.被extern "C"限定的函數(shù)或變量是extern類(lèi)型的;extern是C/C++語(yǔ)言中表明函數(shù)和全局變量作用范圍(可見(jiàn)性)的關(guān)鍵字,該關(guān)鍵字告訴編譯器,其聲明的函數(shù)和變量可以在本模塊或其它模塊中使用。被extern "C"修飾的變量和函數(shù)是按照C語(yǔ)言方式編譯和連接的。
4.與extern對(duì)應(yīng)的關(guān)鍵字是static,被它修飾的全局變量和函數(shù)只能在本模塊中使用。因此,一個(gè)函數(shù)或變量只可能被本模塊使用時(shí),其不可能被extern “C”修飾。
二、extern“C”與__cplusplus
#ifdef __cplusplus
extern "C" {
#endif
#ifdef __cplusplus
}
#endif
Cplusplus(C plus plus)即"C++",用于C++文檔的頭文件中,上面代碼的意思是:如果是C++文件(*.cpp)后綴,則使用extern “C”,在C++項(xiàng)目中應(yīng)用的非常廣泛。即使用gcc編譯器編譯,函數(shù)名為C類(lèi)型如_foo。個(gè)人認(rèn)為,搞懂了這兩個(gè)關(guān)鍵字,尤其是理解extern "C"(再次強(qiáng)調(diào),不為過(guò),呵呵),接下來(lái)的混合編程也就差不多了,哈哈哈。。。。
三、C調(diào)用C++函數(shù)(接口)
1.設(shè)計(jì)程序,共四個(gè)文件
animal.cpp animal.h main.c Makefile
1.1 animal.h
[root@localhost CC++]#cat animal.h
#ifndef __ANIMAL_H__ //防止被重復(fù)包含
#define __ANIMAL_H__
#ifdef __cplusplus
extern "C" {
#endif
class ANIMAL{
public:
ANIMAL(char* );
~ANIMAL();
char* getname(void);
private:
char* name;
};
void print(void);
#ifdef __cplusplus
}
#endif
#endif // __ANIMAL_H__
1.2 animal.cpp:C++文件
[root@localhost CC++]#cat animal.cpp
#include "animal.h"
#include <iostream>
using namespace std;
ANIMAL::ANIMAL(char* data)//構(gòu)造函數(shù)
{ name = new char[64];
strcpy(name, data);
}
ANIMAL::~ANIMAL() //析構(gòu)函數(shù)
{
if(name)
{
delete[] name;
name = NULL;
}
}
char* ANIMAL::getname(void)
{ return name;
}
void print(void) //對(duì)外接口,而且必須有一個(gè)非類(lèi)中方法,才能被C調(diào)用
{
ANIMAL animal("dog");
char* animal_name = animal.getname();
cout << "animal name is :" << animal_name << endl;
}
1.3 main.c:C文件
[root@localhost CC++]#cat main.c
int main(void)
{ print();
return 0;
}
1.4 Makefile
[root@localhost CC++]#cat Makefile
main:main.c animal.o
gcc -lstdc++ main.c animal.o -o main
animal.o:animal.h
g++ -c animal.cpp
.PHONY : clean
clean:
-rm animal.o main
2.測(cè)試
2.1生成可執(zhí)行程序main
[root@localhost CC++]#make
g++ -c animal.cpp
gcc -lstdc++ main.c animal.o -o main
2.2運(yùn)行可執(zhí)行程序main
[root@localhost CC++]# ./main
animal name is :dog
四、C++調(diào)用C函數(shù)
應(yīng)該這個(gè)比較簡(jiǎn)單,我就不多寫(xiě)了,就直接寫(xiě)代碼。
共有三個(gè)文件:1.h 1.c main.cpp
[root@localhost aa]#cat 1.h
#ifndef _1__H_
#define _1__H_
extern void print(char* );
#endif
[root@localhost aa]#cat 1.c
#include <stdio.h>
#include "1.h"
void print(char* data)
{
printf("%s\n", data);
}
[root@localhost aa]#cat main.cpp
extern "C"{
#include "1.h"}
int main(void)
{
print(“hello,world\n”);
return 0;
}
gcc –c 1.c
g++ main.cpp 1.o
接著./a.out,又可以出現(xiàn)我們神奇的hello,world了,C++調(diào)用C代碼很簡(jiǎn)單,但C調(diào)用C++接口可把我給累壞了,苦啊。就是這個(gè)gcc后面跟的-lstdc++害的,出現(xiàn)undefined reference to `__gxx_personality_v0'這個(gè)錯(cuò)誤。是因?yàn)槟阌?SPAN>gcc編譯.cpp文件(animal.cpp).按系統(tǒng)默認(rèn).cpp文件是c++的文件格式。當(dāng)然,混搭時(shí),我還遇到了其他的一些問(wèn)題,都是一些小問(wèn)題,如果上面解釋的還不足以讓你解決C\C++混合編程的問(wèn)題,可以聯(lián)系我哦.
相關(guān)文章
vscode和cmake編譯多個(gè)C++文件的實(shí)現(xiàn)方法
這篇文章主要介紹了vscode和cmake編譯多個(gè)C++文件的實(shí)現(xiàn)方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-03-03C語(yǔ)言結(jié)構(gòu)體計(jì)算內(nèi)存占用問(wèn)題解析
這篇文章主要介紹了C語(yǔ)言結(jié)構(gòu)體計(jì)算內(nèi)存占用問(wèn)題解析,本文通過(guò)案例來(lái)解析了C語(yǔ)言計(jì)算結(jié)構(gòu)體內(nèi)存的方式和方法,需要的朋友可以參考下2021-07-07C++實(shí)現(xiàn)LeetCode(76.最小窗口子串)
這篇文章主要介紹了C++實(shí)現(xiàn)LeetCode(76.最小窗口子串),本篇文章通過(guò)簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-07-07使用C語(yǔ)言來(lái)解決循環(huán)隊(duì)列問(wèn)題的方法
這篇文章主要介紹了使用C語(yǔ)言來(lái)解決循環(huán)隊(duì)列問(wèn)題的方法,來(lái)自ACM的練習(xí)題實(shí)例,需要的朋友可以參考下2015-08-08C語(yǔ)言中進(jìn)程信號(hào)集的相關(guān)操作函數(shù)詳解
這篇文章主要介紹了C語(yǔ)言中進(jìn)程信號(hào)集的相關(guān)操作函數(shù)詳解,包括sigismember函數(shù)和sigfillset函數(shù)以及sigemptyset函數(shù)的用法,需要的朋友可以參考下2015-09-09