当前位置:首页 > 单片机 > C语言与CPP编程
[导读]来源:公众号【编程珠玑】作者:守望先生前言如何在C代码中调用写好的C接口?你可能会奇怪,C不是兼容C吗?直接调用不就可以了?这里我们先按下不表,先看看C如何调用C代码接口。C如何调用C接口为什么会有这样的情况呢?想象一下,有些接口是用C实现的,并提供了库,那么C中该如何使用呢?我...

来源:公众号【编程珠玑】

作者:守望先生

前言

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

C 如何调用C接口

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

//来源:公众号【编程珠玑】// 博客:https://www.yanbinghu.com
//test.c
#include"test.h"
void testCfun()
{
    printf("I am c fun\n");
    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"<<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不一致,所以即便是函数名一样,对于C和C 来说,它们最终的函数签名还是不一样。当然这里又是另外一回事了,我们不细说。我们看看两个文件里的函数符号有什么区别:

$ nm test.o|grep testCfun
0000000000000000 T testCfun
$ nm main.o|grep testCfun
                U _Z8testCfunv
所以它们两个能链接在一起才真是奇怪了呢!名字都不同,还怎么链接?

如何处理

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

#include
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
#ifdef __cplusplus
extern "C"{
#endif

void testCfun();

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

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

来源:公众号【编程珠玑】,专注但不限于分享计算机编程基础,Linux,C语言,C ,数据结构与算法,工具,资源等编程相关[原创]技术文章。博客:https://www.yanbinghu.com

问题

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

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

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

总结

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

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: 驱动电源

在工业自动化蓬勃发展的当下,工业电机作为核心动力设备,其驱动电源的性能直接关系到整个系统的稳定性和可靠性。其中,反电动势抑制与过流保护是驱动电源设计中至关重要的两个环节,集成化方案的设计成为提升电机驱动性能的关键。

关键字: 工业电机 驱动电源

LED 驱动电源作为 LED 照明系统的 “心脏”,其稳定性直接决定了整个照明设备的使用寿命。然而,在实际应用中,LED 驱动电源易损坏的问题却十分常见,不仅增加了维护成本,还影响了用户体验。要解决这一问题,需从设计、生...

关键字: 驱动电源 照明系统 散热

根据LED驱动电源的公式,电感内电流波动大小和电感值成反比,输出纹波和输出电容值成反比。所以加大电感值和输出电容值可以减小纹波。

关键字: LED 设计 驱动电源

电动汽车(EV)作为新能源汽车的重要代表,正逐渐成为全球汽车产业的重要发展方向。电动汽车的核心技术之一是电机驱动控制系统,而绝缘栅双极型晶体管(IGBT)作为电机驱动系统中的关键元件,其性能直接影响到电动汽车的动力性能和...

关键字: 电动汽车 新能源 驱动电源

在现代城市建设中,街道及停车场照明作为基础设施的重要组成部分,其质量和效率直接关系到城市的公共安全、居民生活质量和能源利用效率。随着科技的进步,高亮度白光发光二极管(LED)因其独特的优势逐渐取代传统光源,成为大功率区域...

关键字: 发光二极管 驱动电源 LED

LED通用照明设计工程师会遇到许多挑战,如功率密度、功率因数校正(PFC)、空间受限和可靠性等。

关键字: LED 驱动电源 功率因数校正

在LED照明技术日益普及的今天,LED驱动电源的电磁干扰(EMI)问题成为了一个不可忽视的挑战。电磁干扰不仅会影响LED灯具的正常工作,还可能对周围电子设备造成不利影响,甚至引发系统故障。因此,采取有效的硬件措施来解决L...

关键字: LED照明技术 电磁干扰 驱动电源

开关电源具有效率高的特性,而且开关电源的变压器体积比串联稳压型电源的要小得多,电源电路比较整洁,整机重量也有所下降,所以,现在的LED驱动电源

关键字: LED 驱动电源 开关电源

LED驱动电源是把电源供应转换为特定的电压电流以驱动LED发光的电压转换器,通常情况下:LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: LED 隧道灯 驱动电源
关闭