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

作者: 罗蓁蓁 | 来源:发表于2021-11-08 10:49 被阅读0次

✨有的时候,当我想在c++代码中调用c接口,然后就编译报错了!!!

引出问题

你可能会奇怪,C++不是兼容C吗?直接调用不就可以了?为什么会有这样的情况呢?想象一下,有些接口是用C实现的,并提供了库,那么C++中该如何使用呢?

我们先不做任何区别对待,看看普通情况下会发生什么意想不到的事情。首先提供一个C接口:

//test.c
#include"test.h"
void testCfun()
{
    printf("I am c fun\n");
    return;
}

我们在这里编译成C目标文件:

gcc -c test.c

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

#include<stdio.h>
void testCfun();

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

//main.cpp
#include"test.h"
#include<iostream>
using namespace std;
int main(void)
{
    /*调用C接口*/
    cout<<"start to call c function"<<endl;
    testCfun();
    cout<<"end to call c function"<<endl;
    return 0;
}

编译:

$ 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并不支持。C++为了支持函数重载,它在“生成”函数符号信息时,不能仅仅通过函数名,因为重载函数的函数名都是一样的,所以它还要根据入参,命名空间等信息来确定唯一的函数签名。

或者说C++生成函数签名的方式与C不一致,所以即便是函数名一样,对于C和C++来说,它们最终的函数签名还是不一样。当然这里又是另外一回事了,我们不细说。我们看看两个文件里的函数符号有什么区别:

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

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

解决方法

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

#include<stdio.h>
extern "C"{
void testCfun();
}

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

$ g++ -o main main.cpp test.o
$ ./main
start to call c function
I am c fun
end to call c function

优化

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

//main.c
#include"test.h"
int main(void)
{
    /*调用C接口*/
    testCfun();
    return 0;
}

编译:

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

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

#include<stdio.h>
#ifdef __cplusplus
extern "C"{
#endif

void testCfun();

#ifdef __cplusplus
}
#endif

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

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

总结

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

记住,c代码被c++调用,是需要用extern "C"把被调用的函数的声明给括起来。

关注&&联系

gitee: https://gitee.com/cmcc-oneos/OneOS-Lite

docs: https://oneos-lite.com/

引用自:https://gitee.com/cmcc-oneos/OneOS-Lite/blob/dev/docs/quick_guide/cplusplus/c++-call-c.md

相关文章

  • Go调用C/C++

    cgo golang是类C的语言 支持调用C接口(不支持调用C++)Go调用C/C++的方式 : C : 直接调用...

  • 各种加密算法编程演示实验

    实验目的 掌握各种对称算法接口的调用方法(C/C++方式) 掌握各种非对称算法接口的调用方法(C/C++方式) 掌...

  • C/C++与Java之间的互相调用

    利用JNI技术,可以做到C/C++与Java之间的相互调用。 1、Java调用C/C++接口 一般把C/C++写成...

  • Go与c++混合编程

    利用cgo实现go调用c/c++的接口

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

    ✨有的时候,当我想在c++代码中调用c接口,然后就编译报错了!!! 引出问题 你可能会奇怪,C++不是兼容C吗?直...

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

    ✨有的时候,当我想在c代码中调用c++接口,然后就编译报错了!!! 引出问题 C++支持函数重载的, 函数名相同但...

  • C++调用lua方式

    目标 使用C++调用lua接口 示例 lua代码(test.lua) C++调用示例(lua_test.cpp) ...

  • 「转」C++ 和 JS 交互

    本章主要来讲讲如何通过 V8 来实现 JS 调用 C++。JS 调用 C++,分为 JS 调用 C++ 函数(全局...

  • C++子线程调用Java方法

    1. C++ 全局调用Java方法 之前讨论过,如何C++主线程中调用 Java 函数C++主线程调用Java方法...

  • Fortran与C++交叉编程:Fortran如何调用C++动态

    Fortran程序中调用C++编写的动态库的问题分为两个子问题 Fortran调用c形式函数接口 为c++动态库编...

网友评论

    本文标题:C++是如何调用C接口的?

    本文链接:https://www.haomeiwen.com/subject/ohjuzltx.html