溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點(diǎn)擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

C 與 C++ 接口函數(shù)相互調(diào)用的實(shí)現(xiàn)

發(fā)布時間:2020-10-13 13:05:19 來源:腳本之家 閱讀:134 作者:云亭 欄目:編程語言

一、C 或 C++ 編譯的四個步驟

(一) 預(yù)處理

在該步驟中,編譯器將源程序中以“#”開頭的語句進(jìn)行處理。其中,#include 的原理是將目標(biāo)文件內(nèi)容導(dǎo)入本文件。

(二) 編譯

在該步驟中,編譯器將第一步生成的各個文件分別轉(zhuǎn)換成匯編語言文件。在該過程中,所有函數(shù)的名稱都會被轉(zhuǎn)換成一個符號作為匯編文件中的唯一標(biāo)識,對 C 語言函數(shù)一般直接用函數(shù)名稱作為其唯一標(biāo)識的符號,而對于 C++ 函數(shù)在多數(shù)情況下需要在函數(shù)名稱加上各種前綴或后綴才能作為其標(biāo)識,比如函數(shù) void Print(int num),如果編譯器將其視為 C 語言編譯,則該函數(shù)在匯編文件中的符號為 Print,若視為 C++,則其符號可能為 Print_int(在 gcc 或 g++ 中函數(shù)名稱的改變還會考慮命名空間等因素),這也是 C++ 支持函數(shù)重載的原因。

(三) 匯編

在該步驟中,編譯器將第二步生成的各個文件分別轉(zhuǎn)換為二進(jìn)制文件,但還不是可執(zhí)行文件。

(四) 鏈接

在該步驟中,編譯器會為第三步生成的每一個文件“穿針引線”,比如 main() 函數(shù)中調(diào)用了 Print() 函數(shù),還不知道 Print() 函數(shù)在哪里,而在 Print() 函數(shù)主體所在的那個文件中,已經(jīng)標(biāo)明了 Print() 函數(shù)的地址,所以編譯器會在 main() 函數(shù)中調(diào)用 Print() 函數(shù)的地方標(biāo)注 Print() 函數(shù)的地址,為程序執(zhí)行過程中的地址跳轉(zhuǎn)提供目標(biāo)地址,而編譯器能做到這一步的前提,是 main() 函數(shù)中 Print() 函數(shù)的標(biāo)識,和 Print() 函數(shù)主體所在的那個文件中 Print() 函數(shù)的標(biāo)識是一模一樣的,如果不一樣,就會觸發(fā)鏈接錯誤。

二、C 與 C++ 接口相互調(diào)用的關(guān)鍵

從上文可以得知,要調(diào)用一個函數(shù)有一個重要條件就是調(diào)用處的符號和函數(shù)主體處的符號要一模一樣,而 C 和 C++ 在編譯過程中將函數(shù)名稱改編成標(biāo)識符號的方法是不一樣的,因此相互調(diào)用的關(guān)鍵在于統(tǒng)一接口函數(shù)的標(biāo)識符號,而一般采取的方法是,用 C 函數(shù)改編的方法統(tǒng)一接口函數(shù)的改編方式。

三、extern "C"

extern "C" 的作用是告訴編譯器按 C 函數(shù)名稱改編的方法將修飾的函數(shù)改編成標(biāo)識符號。extern "C" 一般用在 C++ 文件中。

extern "C" void Print(int num);
extern "C" {
 void Input(int* num);
 void Output(int num);
};

以上是 extern "C" 的兩種寫法。如此一來,以上三個函數(shù)都會按 C 的方式被改編成符號,在 gcc 或 g++ 編譯下就會被改變成 Print,Input,Output。

四、C 函數(shù)調(diào)用 C++ 接口

(一) 調(diào)用非成員函數(shù)

被調(diào)用函數(shù)的聲明和定義如下。

/**
 * called.h
 */
#ifndef CALLED_H
#define CALLED_H

extern "C" void PrintCpp(void);

#endif

/**
 * called.cpp
 */
#include <iostream>
#include "called.h"

using namespace std;

void
PrintCpp(void) {
 cout << "I\'m cpp." << endl;
}

最終調(diào)用如下。

/**
 * call.c
 */
#include "called.h"

int
main(int argc, char const* argv[]) {
 PrintCpp();
 return 0;
}

(二) 調(diào)用類成員函數(shù)(接口函數(shù)沒有類指針)

被調(diào)用函數(shù)聲明和定義如下。

/**
 * called.h
 */
#ifndef CALLED_H
#define CALLED_H

class Console {
public:
 Console();
 virtual void PrintDouble(double num);
};

extern "C" void CppPrintDouble(double num);

#endif
/**
 * called.cpp
 */
#include <iostream>
#include "called.h"

using namespace std;

Console::Console() {}

void
Console::PrintDouble(double num) {
 cout << num << endl;
}

Console* console = new Console();

void
CppPrintDouble(double num) {
 console->PrintDouble(num);
}

最終調(diào)用如下。

/**
 * call.c
 */
#include "called.h"

int
main(int argc, char const* argv[]) {
 CppPrintDouble(3.14);
 return 0;
}

五、C++ 函數(shù)調(diào)用 C 接口

被調(diào)用函數(shù)的聲明和定義如下。

/**
 * called.h
 */
#ifndef CALLED_H
#define CALLED_H

void PrintC(void);

#endif
/**
 * called.c
 */
#include <stdio.h>

#ifdef __cplusplus
extern "C" {
#endif
#include "called.h"
#ifdef __cplusplus
};
#endif

void
PrintC(void) {
 printf("I\'m C.\n");
}

最終調(diào)用如下。

/**
 * call.cpp
 */
#ifdef __cplusplus
extern "C" {
#endif
#include "called.h"
#ifdef __cplusplus
};
#endif

int
main(int argc, char const* argv[]) {
 PrintC();
 return 0;
}

在 called.c 文件中,#ifdef __cplusplus /*...*/ #endif 和 extern "C" 的作用是防止 g++ 編譯器對“.c”文件用 C++ 的方式編譯,如果用 gcc 進(jìn)行編譯,則直接寫 #include "called.h" 就行。

到此這篇關(guān)于C 與 C++ 接口函數(shù)相互調(diào)用的實(shí)現(xiàn)的文章就介紹到這了,更多相關(guān)C 與 C++ 接口函數(shù)調(diào)用內(nèi)容請搜索億速云以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持億速云!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI