这篇文章主要介绍了C++怎么调用已经写好的C接口,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。
一个C语言文件test.c
#include <stdio.h> void print(int a,int b) { printf("这里调用的是C语言的函数:%d,%d\n",a,b); }
一个头文件test.h
#ifndef _TEST_H #define _TEST_H void print(int a,int b); #endif
C++文件调用C函数
#include <iostream> using namespace std; #include "test.h" int main() { cout<<"现在调用C语言函数\n"; print(3,4); return 0; }
执行命令
gcc -c test.c g++ -o main main.cpp test.o
编译后链接出错:main.cpp
对print(int, int)
未定义的引用。
那么g++编译器为什么找不到print(int,int)
呢,其实在我们学C++重载的时候就提到过C++底层的编译原理。
test.c
我们使用的是C语言的编译器gcc进行编译的,其中的函数print
编译之后,在符号表中的名字为 print
,通过nm查看.o文件.
$ gcc -c test.c $ nm test.o U _GLOBAL_OFFSET_TABLE_ 0000000000000000 T print U printf
我们链接的时候采用的是 g++ 进行链接,也就是 C++ 链接方式,程序在运行到调用 print
函数的代码时,会在符号表中寻找 _Z5printii
(是按照C++的链接方法来寻找的,所以是找 _Z5printii
而不是找 print
)的名字,发现找不到,所以会提示“未定义的引用”
$ 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语言是 _print
,C++
是 __Z5printii
修改test.h文件
#ifndef _TEST_H #define _TEST_H extern "C"{ void print(int a,int b); } #endif
修改后再次执行命令
gcc -c test.c g++ -o main main.cpp test.o ./main
运行无报错
实验:定义main.c
函数如下
#include <stdio.h> #include "test.h" int main() { printf("现在调用C语言函数\n"); print(3,4); return 0; }
重新执行命令如下
gcc -c test.c gcc -o mian main.c test.o
报错:C语言里面没有extern “C“这种写法
为了使得test.c
代码既能被C++调用又能被C调用
将test.h修改如下
#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
的插件,快速生成上面的代码
再次执行命令
gcc -c test.c gcc -o main main.c test.o ./main
结果示意:
感谢你能够认真阅读完这篇文章,希望小编分享的“C++怎么调用已经写好的C接口”这篇文章对大家有帮助,同时也希望大家多多支持亿速云,关注亿速云行业资讯频道,更多相关知识等着你来学习!
免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。