- 前言
- C++调用C文件
- 原因分析
- 总结
- 解决调用失败问题
- 思考:那C文件能够调用C函数吗
- C代码既能被C++调用又能被C调用
如何在C++代码中调用写好的C接口?你可能会奇怪,C++不是兼容C吗?直接调用不就可以了,那么我们来测试一下,先看看C++如何调用C代码接口的。
C++调用C文件一个C语言文件test.c
#includevoid print(int a,int b) { printf("这里调用的是C语言的函数:%d,%dn",a,b); }
一个头文件test.h
#ifndef _TEST_H #define _TEST_H void print(int a,int b); #endif
C++文件调用C函数
#includeusing 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文件.
ubuntu@VM-16-5-ubuntu:~/dengzr/c++$ gcc -c test.c
ubuntu@VM-16-5-ubuntu:~/dengzr/c++$ nm test.o
U _GLOBAL_OFFSET_TABLE_
0000000000000000 T print
U printf
我们链接的时候采用的是 g++ 进行链接,也就是 C++ 链接方式,程序在运行到调用 print 函数的代码时,会在符号表中寻找 _Z5printii(是按照C++的链接方法来寻找的,所以是找 _Z5printii 而不是找 print)的名字,发现找不到,所以会提示“未定义的引用”
ubuntu@VM-16-5-ubuntu:~/dengzr/c++$ g++ -c test.c
ubuntu@VM-16-5-ubuntu:~/dengzr/c++$ ls
main.cpp makefile test.c test.h test.o
ubuntu@VM-16-5-ubuntu:~/dengzr/c++$ 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
运行无报错
思考:那C文件能够调用C函数吗实验:定义main.c函数如下
#include#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“这种写法
C代码既能被C++调用又能被C调用为了使得test.c代码既能被C++调用又能被C调用
将test.h修改如下
#ifndef __TEST_H__
#define __TEST_H__
#ifdef __cplusplus
#if __cplusplus
extern "C"{
#endif
#endif
extern void print(int a,int b);
#ifdef __cplusplus
#if __cplusplus
}
#endif
#endif
#endif
ps:下期介绍一个Source Insight的插件,快速生成上面的代码
再次执行命令
gcc -c test.c gcc -o main main.c test.o ./main
结果示意:



