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

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

1. C++ 如何调用 C 接口

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

首先提供一个 C 接口:

// 来源:公众号【编程珠玑】博客:https://www.yanbinghu.com
//test.c
#include"test.h"
void testCfun()
{printf("I am c fun");
    return;
}

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

gcc -c test.c

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

#include
void testCfun();

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

// 来源:公众号【编程珠玑】博客:https://www.yanbinghu.com
//main.cpp
#include"test.h"
#include
using namespace std;
int main(void)
{
    /* 调用 C 接口 */
    cout<<"start to call c function"<

编译:

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

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

2. 真相

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

$ nm test.o|grep testCfun
0000000000000000 T testCfun
$ nm main.o|grep testCfun
                U _Z8testCfunv

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

3. 如何处理

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

  • 分享于 · 2019.08.29 16:22 · 阅读 · 2511

[版权声明] :本文系网友分享,仅以非商业性的交流和科研为目的,并不代表本网赞同其观点和对其真实性负责。如涉及作品内容、版权和其它问题,请在30日内与本网( friends@stuch.cn )联系!我们将协调给予处理。转载授权码:确权中,请原作者尽快与我们取得联系,阅读原文(请登录)..

0 条评论

请先 登录 后评论
猜猜我是谁
孟云

10
提问
2
回答
5
文章