背景
在c与C++编程中难免两者混用,而c与c++特性的不同,其实这其中有很多坑。比如c其实是不支持同名函数重载的。
如在c中写下如下两个函数:
void test()
{
}
void test(int a)
{
}
编译时报出以下错误:
技术描述
c++ 之所以支持函数重载,即通过函数的参数不同可以区分不同的函数主体,实际是因为在编译后的调用约定不同,c++编译后由函数名和参数共同指代一个函数主体。可借鉴map由于函数名和参数共同作为key来索引函数主体。
而在编码时,extern"C" 便是用于处理c++与c混用的情况。即声明该函数为c函数,因此添加此关键字后函数不能再有同名函数存在。
extern "C" 写在哪里
新手都知道用extern "C" ,但是不知道将extern "C"写在哪里。其实只需要记住一条准则,即包含关系准则。而不去区分声明和定义。
当写在.h中,至少在被使用的.c或cpp 以及实现函数的c或cpp中出现。因此该情况等效于在实现和.h中均写上 extern "C" 。出现错误时也可以用这样的办法去排查。
除开.h的位置则写在哪里就是仅仅哪里。所以如果写在实现函数的地方,且不是在.h中,那必然需要在其他地方也写上。但是如果为c文件,其中是无extern"C"关键字支持的,因此最合理的方式还是在.h中写,但是使用宏进行隔离。
extern "C" 使用准则,解决错误
只需要将.h声明加上宏,如下代码段
c使用CPP中函数,在.h声明前写上 extern "C" 让其成为一个简单的 c函数
cpp中使用c函数,在.h中声声明前写,但是由于c中无该关键字,使用宏隔离
//.h文件
#ifdef __cplusplus
extern "C" {
#endif
//函数xxx
#ifdef __cplusplus
}
#endif
在c编译时,宏不生效。.h和c文件一致,都是c的格式。
在编译使用该函数的cpp时,宏生效,.h声明指明为c函数,和obj一致。
剖析错误出现原因
上面的分析我没有细节的把.h情况拆开,但是可以过一遍编译的过程。
不添加extern "C" 关键字情况
当c文件被编译时,.h和c一致,编译为c函数二进制
当使用函数的cpp被编译时,.h无特殊声明,当做cpp处理,查找c++的函数,无法找到出错
在.h添加extern "C" 关键字情况
当c文件被编译时,.h和c不一致,.h出现extern 未标识关键字,出错
当使用函数的cpp被编译时,.h有声明,当做c函数处理,但是c编译出错,也无法找到、
在.h添加extern "C" 且用宏处理
看上面的分析