文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

C++中是如何调用C接口的?

2024-12-11 17:21

关注

前言

如何在C++代码中调用写好的C接口?你可能会奇怪,C++不是兼容C吗?直接调用不就可以了?这里我们先按下不表,先看看C++如何调用C代码接口。

C++如何调用C接口

为什么会有这样的情况呢?想象一下,有些接口是用C实现的,并提供了库,那么C++中该如何使用呢?我们先不做任何区别对待,看看普通情况下会发生什么意想不到的事情。

首先提供一个C接口:

 

  1. //来源:公众号【编程珠玑】 
  2. // 博客:https://www.yanbinghu.com 
  3. //test.c 
  4. #include"test.h" 
  5. void testCfun() 
  6.     printf("I am c fun\n"); 
  7.     return

为了简化,我们在这里就不将它做成静态库或者动态库了,有兴趣的可以参考《静态库制作》自行尝试。我们在这里编译成C目标文件:

 

  1. gcc -c test.c 

另外提供一个头文件test.h:

 

  1. #include 
  2. void testCfun(); 

我们的C++代码调用如下:

 

  1. //来源:公众号【编程珠玑】 博客:https://www.yanbinghu.com 
  2. //main.cpp 
  3. #include"test.h" 
  4. #include 
  5. using namespace std; 
  6. int main(void) 
  7.      
  8.     cout<<"start to call c function"<
  9.     testCfun(); 
  10.     cout<<"end to call c function"<
  11.     return 0; 

编译:

 

  1. $ g++ -o main main.cpp test.o 
  2. /tmp/ccmwVJqM.o: In function `main': 
  3. main.cpp:(.text+0x21): undefined reference to `testCfun()' 
  4. collect2: error: ld returned 1 exit status 

很不幸,最后的链接报错了,说找不到testCfun,但是我们确实定义了这个函数。为什么会找不到呢?现在你还会认为C++直接就可以调用C接口了吗?

真相

我们都知道,C++中函数支持重载,而C并不支持。C++为了支持函数重载,它在“生成”函数符号信息时,不能仅仅通过函数名,因为重载函数的函数名都是一样的,所以它还要根据入参,命名空间等信息来确定唯一的函数签名。或者说C++生成函数签名的方式与C不一致,所以即便是函数名一样,对于C和C++来说,它们最终的函数签名还是不一样。当然这里又是另外一回事了,我们不细说。我们看看两个文件里的函数符号有什么区别:

 

  1. $ nm test.o|grep testCfun 
  2. 0000000000000000 T testCfun 
  3. $ nm main.o|grep testCfun 
  4.                 U _Z8testCfunv 

所以它们两个能链接在一起才真是奇怪了呢!名字都不同,还怎么链接?

如何处理

那么如何处理呢?很显然,我们必须告诉链接器,这是一个C接口,而不是C++接口,所以需要加入 extern C,我们修改test.h

 

  1. #include 
  2. extern "C"
  3. void testCfun(); 

这里用extern "C"将testCfun接口包裹起来,告诉编译器,这里的是C代码哈,你要按C代码的方式处理。再次编译:

 

  1. $ g++ -o main main.cpp test.o 
  2. $ ./main 
  3. start to call c function 
  4. I am c fun 
  5. end to call c function 

看终端输出,完美!

优化

虽然上面的C接口可以被C++正常调用了,但是如果这个C接口要被C代码调用呢?增加main.c内容如下

 

  1. //main.c 
  2. #include"test.h" 
  3. int main(void) 
  4.      
  5.     testCfun(); 
  6.     return 0; 

编译:

 

  1. $ gcc -o main main.c test.c 
  2. In file included from main.c:2:0: 
  3. test.h:2:8: error: expected identifier or '(' before string constant 
  4.  extern "C"
  5.         ^ 
  6. In file included from test.c:2:0: 
  7. test.h:2:8: error: expected identifier or '(' before string constant 
  8.  extern "C"

不出意外,又报错了,很显然,C语言中并没有extern "C"这样的写法,所以为了能使得test.c的代码既能被C++调用,也能被C调用,需要改写成下面这样:

 

  1. #include 
  2. #ifdef __cplusplus 
  3. extern "C"
  4. #endif 
  5.  
  6. void testCfun(); 
  7.  
  8. #ifdef __cplusplus 
  9. #endif 

这里通过__cplusplus宏来控制是否需要extern “C”,如果是C++编译器,那么extern "C"部分就会被预处理进去,这样test.c代码就可以既用于C++,也可以用于C啦。

赶快去你的C项目代码头文件中看看,是不是也有这样的代码段呢?

问题

为什么我们在C++代码中可以直接调用一些标准C库函数呢?即使你在main函数中调用printf等函数,它也不会出现链接错误。因为库函数已经有了类似的处理了。

如果你还是不确定,你可以先预处理:

 

  1. $ g++ -E main.i main.cpp 

去生成的main.i文件中找一找,是不是有extern "C"。

总结

C++支持重载,而C不支持,C++并不能直接调用C代码写好的接口,因此如果你的C代码想要能够被C调用,也想被C++调用,那么别忘了extern "C"。

那么问题来了,C又该如何调用C++的接口呢?

来源:C语言与C++编程内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯