经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » 程序设计 » C 语言 » 查看文章
C与C++的互相调用!就像大学宿舍一样,我用你的,你用我的!
来源:cnblogs  作者:C语言编程学习基地  时间:2020/11/23 12:21:47  对本文有异议

    注意,本文的前提是,c代码采用gcc等c语言编译器编译c代码,采用g++等c++编译器编译c++代码,如果c和c++代码统一使用g++编译,大部分情况是可以实现两者代码相互调用的。

        以下为踩坑过程的总结o_O||。

 

C 与 C++ 的函数区别

        要了解两者之间如何实现相互调用,必须先了解c与c++之间的函数有什么不同。

        c++作为c语言的升级版,两者必然有很多不同之处。

        其中有一个重大不同点就是,c++支持函数重载,而c语言不支持。为了使函数支持重载,c++在c语言的基础上,将函数名添加上返回值和参数的类型信息。

例如,int add(int, int)这个函数,通过c++编译器编译后,可能呈现的函数名为int int_add_int_int(int, int)

(注:此处为大概地说明c++是如何将返回值和参数信息添加到函数名中的,实际中编译器不一定是这样实现的)。

        从以上说明可以得出,由于c++对函数重载的支持,使得编译后的函数符号与c语言的不一致,即使是在两者函数名相同的前提下。

 

extern "C"的作用

        那么,c与c++是不能相互调用了吗?答案是否定的,因为存在着extern "C"这个关键字可以使语句可以按照类C的编译和连接规约来编译和连接,而不是C++的编译的连接规约。这样在类C的代码中就可以调用C++的函数or变量等。

        注意:extern "C"指令中的"C",表示的一种编译和连接规约,而不是一种语言。"C"表示符合C语言的编译和连接规约的任何语言,如Fortran、assembler等。

        还有要说明的是,extern "C"指令仅指定编译和连接规约,但不影响语义。例如在函数声明中,指定了extern "C",仍然要遵守C++的类型检测、参数转换规则。

 

C++ 中调用 C 代码

        对于c++,由于c++的编译器对c语言兼容,因此在c++中调用c语言编写的函数,只需要在函数声明前面加上关键字extern "C",表示采用类c语言的方式解析函数符号。例子如下:

  1. // add.h
  2. #ifdef __ADD_H__
  3. #define __ADD_H__
  4.  
  5. extern "C" int add(int a, int b);
  6. #endif
  7.  
  8. // add.c
  9.  
  10. int add(int a, int b)
  11. {
  12. return a + b;
  13. }
  14. // main.cc
  15. #include <iostream>
  16. #include "add.h"
  17.  
  18. using namespace std;
  19. int main()
  20. {
  21. cout << "1 + 1 = " << add(1, 1) << endl;
  22. }

 

        在例子中,main.cc为c++代码,add.c为c语言代码,当c++编译器识别到extern "C"`关键字时,会去寻找add函数的实现而不是寻找类似int_add_int_int这样带参数信息的函数实现。

 

C 语言调用 C++ 代码

        c语言调用c++代码却并不容易,原因是c语言并不兼容c++。

        就算c语言可以调用c++,也会因为无法识别c++新定义的符号而编译报错。

        因此,为了实现c语言调用c++函数,必须实现以下两个步骤:

1. 将c++相关函数封装为静态库或动态库(因为调用库函数时编译器并不知道里面执行的是什么语言);

2. 对外提供遵循类c语言规约的接口函数。

 

        例子如下所示:

  1. // printNum.h
  2. #ifdef __PRINTNUM_H__
  3. #define __PRINTNUM_H__
  4.  
  5. extern "C" void printNum(int a);
  6. #endif
  7.  
  8. // printNum.cc
  9. #include <iostream>
  10. #include "printNum.h"
  11.  
  12. using namespace std;
  13. void printNum(int a)
  14. {
  15. cout << << "num is " << a << endl;
  16. }
  17. // main.c
  18.  
  19. extern void printNum(int a);
  20. printNum(5);

 

        通过将cout函数封装为类c语言规约的接口函数,使得main.c中可以成功调用c++函数printNum

        值得注意的是,main.c不可以直接引入printNum.h,因为c语言不能识别extern "C"关键字。可以利用c++预定义宏实现头文件的改写:

  1. #ifdef __PRINTNUM_H__
  2. #define __PRINTNUM_H__
  3. #ifdef __cplusplus
  4. extern "C" {
  5. #endif
  6.  
  7. void printNum(int a);
  8. #ifdef __cplusplus
  9. }
  10. #endif
  11.  
  12. #endif
  1. #ifdef __PRINTNUM_H__
  2. #define __PRINTNUM_H__
  3. #ifdef __cplusplus
  4. extern "C" {
  5. #endif
  6.  
  7. void printNum(int a);
  8. #ifdef __cplusplus
  9. }
  10. #endif
  11.  
  12. #endif

 

 

小结

    ? c语言与c++的相互调用可以通过 extern "C" 关键字实现

    ? c++中调用c代码,只须在c++中为c代码函数声明之前加上extern "C"

    ? c语言调用c++代码,则需要将c++代码编译成静态库或动态库,然后对外提供用 extern "C" 声明的类c封装函数

 


 

不管你是转行也好,初学也罢,进阶也可——值得关注进入】的C/C++编程学习进阶俱乐部

涉及到:C语言、C++、windows编程、网络编程、QT界面开发、Linux编程、游戏编程、黑客等等......


 

一个活跃、高格调、高层次的程序员编程学习殿堂;编程入门只是顺带,思维的提高才有价值!

原文链接:http://www.cnblogs.com/huya-edu/p/14005931.html

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号