我是靠谱客的博主 清爽火,这篇文章主要介绍C++如何调用已经写好的C接口,现在分享给大家,希望可以做个参考。

前言:

如何在C++代码中调用写好的C接口?你可能会奇怪,C++不是兼容C吗?直接调用不就可以了,那么我们来测试一下,先看看C++如何调用C代码接口的。

1、C++调用C文件

一个C语言文件test.c

复制代码
1
2
3
4
5
#include <stdio.h> void print(int a,int b) { printf("这里调用的是C语言的函数:%d,%dn",a,b); }

一个头文件test.h

复制代码
1
2
3
4
5
6
#ifndef _TEST_H #define _TEST_H void print(int a,int b); #endif

C++文件调用C函数

复制代码
1
2
3
4
5
6
7
8
9
#include <iostream> using namespace std; #include "test.h" int main() { cout<<"现在调用C语言函数n"; print(3,4); return 0; }

执行命令

复制代码
1
2
gcc -c test.c g++ -o main main.cpp test.o

编译后链接出错:main.cppprint(int, int)未定义的引用。

那么g++编译器为什么找不到print(int,int)呢,其实在我们学C++重载的时候就提到过C++底层的编译原理。

2、原因分析

test.c我们使用的是C语言的编译器gcc进行编译的,其中的函数print编译之后,在符号表中的名字为 print,通过nm查看.o文件.

复制代码
1
2
3
4
5
$ gcc -c test.c $ nm test.o U _GLOBAL_OFFSET_TABLE_ 0000000000000000 T print U printf

我们链接的时候采用的是 g++ 进行链接,也就是 C++ 链接方式,程序在运行到调用 print 函数的代码时,会在符号表中寻找 _Z5printii(是按照C++的链接方法来寻找的,所以是找 _Z5printii 而不是找 print)的名字,发现找不到,所以会提示“未定义的引用”

复制代码
1
2
3
4
5
6
7
$ g++ -c test.c $ ls main.cpp makefile test.c test.h test.o $ nm test.o U _GLOBAL_OFFSET_TABLE_ U printf 0000000000000000 T _Z5printii

此时如果我们在对print的声明中加入 extern “C” ,这个时候,g++编译器就会按照C语言的链接方式进行寻找,也就是在符号表中寻找print(这才是C++兼容C),这个时候是可以找到的,是不会报错的。

总结:

编译后底层解析的符号不同,C语言是 _printC++ __Z5printii

3、解决调用失败问题

修改test.h文件

复制代码
1
2
3
4
5
6
#ifndef _TEST_H #define _TEST_H extern "C"{ void print(int a,int b); } #endif

修改后再次执行命令

复制代码
1
2
3
gcc -c test.c g++ -o main main.cpp test.o ./main

运行无报错

4、思考:那C语言能够调用C接口吗

实验:定义main.c函数如下

复制代码
1
2
3
4
5
6
7
8
#include <stdio.h> #include "test.h" int main() { printf("现在调用C语言函数n"); print(3,4); return 0; }

重新执行命令如下

复制代码
1
2
gcc -c test.c gcc -o mian main.c test.o

报错:C语言里面没有extern “C“这种写法

5、C接口既能被C++调用又能被C调用

为了使得test.c代码既能被C++调用又能被C调用

将test.h修改如下

复制代码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
#ifndef __TEST_H__ #define __TEST_H__ #ifdef __cplusplus #if __cplusplus extern "C"{ #endif #endif /* __cplusplus */ extern void print(int a,int b); #ifdef __cplusplus #if __cplusplus } #endif #endif /* __cplusplus */ #endif /* __TEST_H__ */

ps:下期介绍一个Source Insight的插件,快速生成上面的代码

再次执行命令

复制代码
1
2
3
gcc -c test.c gcc -o main main.c test.o ./main

结果示意:

到此这篇关于C++如何调用已经写好的C接口的文章就介绍到这了,更多相关C++如何调用C接口内容请搜索靠谱客以前的文章或继续浏览下面的相关文章希望大家以后多多支持靠谱客!

最后

以上就是清爽火最近收集整理的关于C++如何调用已经写好的C接口的全部内容,更多相关C++如何调用已经写好内容请搜索靠谱客的其他文章。

本图文内容来源于网友提供,作为学习参考使用,或来自网络收集整理,版权属于原作者所有。
点赞(140)

评论列表共有 0 条评论

立即
投稿
返回
顶部