将外部“C”放入C++代码中具体做什么?
例如:
extern "C" {
void foo();
}
将外部“C”放入C++代码中具体做什么?
例如:
extern "C" {
void foo();
}
当前回答
extern“C”使C++中的函数名具有C链接(编译器不会损坏该名称),以便客户端C代码可以使用仅包含函数声明的C兼容头文件链接到(使用)您的函数。函数定义包含在二进制格式(由C++编译器编译)中,然后客户端C链接器将使用C名称链接到该格式。
由于C++有函数名的重载,而C没有,所以C++编译器不能仅将函数名用作链接到的唯一id,因此它通过添加有关参数的信息来破坏名称。C编译器不需要更改名称,因为您不能重载C中的函数名。当您声明函数在C++中具有外部“C”链接时,C++编译器不会将参数/参数类型信息添加到用于链接的名称中。
正如您所知,您可以显式指定每个单独声明/定义的外部“C”链接,或使用块将声明/定义序列分组以具有特定链接:
extern "C" void foo(int);
extern "C"
{
void g(char);
int i;
}
如果您关心技术细节,请参见C++03标准第7.5节,这里有一个简短的总结(重点是外部“C”):
外部“C”是链接规范每个编译器都需要提供“C”链接链接规范只能出现在命名空间范围内所有函数类型、函数名和变量名都有语言链接参见Richard的评论:只有具有外部链接的函数名和函数名才有语言链接具有不同语言链接的两种函数类型是不同的类型,即使在其他方面相同连杆规格嵌套,内部规格决定最终连杆类成员忽略外部“C”最多一个具有特定名称的函数可以具有“C”链接(无论名称空间如何)外部“C”强制一个函数具有外部链接(不能使其成为静态的)参见Richard的评论:外部内部的静态“C”是有效的;这样声明的实体具有内部链接,因此没有语言链接从C++到用其他语言定义的对象以及从其他语言到用C++定义的对象的链接是由实现定义的,并且依赖于语言。只有当两种语言实现的对象布局策略足够相似时,才能实现这种链接
其他回答
extern“C”使C++中的函数名具有C链接(编译器不会损坏该名称),以便客户端C代码可以使用仅包含函数声明的C兼容头文件链接到(使用)您的函数。函数定义包含在二进制格式(由C++编译器编译)中,然后客户端C链接器将使用C名称链接到该格式。
由于C++有函数名的重载,而C没有,所以C++编译器不能仅将函数名用作链接到的唯一id,因此它通过添加有关参数的信息来破坏名称。C编译器不需要更改名称,因为您不能重载C中的函数名。当您声明函数在C++中具有外部“C”链接时,C++编译器不会将参数/参数类型信息添加到用于链接的名称中。
正如您所知,您可以显式指定每个单独声明/定义的外部“C”链接,或使用块将声明/定义序列分组以具有特定链接:
extern "C" void foo(int);
extern "C"
{
void g(char);
int i;
}
如果您关心技术细节,请参见C++03标准第7.5节,这里有一个简短的总结(重点是外部“C”):
外部“C”是链接规范每个编译器都需要提供“C”链接链接规范只能出现在命名空间范围内所有函数类型、函数名和变量名都有语言链接参见Richard的评论:只有具有外部链接的函数名和函数名才有语言链接具有不同语言链接的两种函数类型是不同的类型,即使在其他方面相同连杆规格嵌套,内部规格决定最终连杆类成员忽略外部“C”最多一个具有特定名称的函数可以具有“C”链接(无论名称空间如何)外部“C”强制一个函数具有外部链接(不能使其成为静态的)参见Richard的评论:外部内部的静态“C”是有效的;这样声明的实体具有内部链接,因此没有语言链接从C++到用其他语言定义的对象以及从其他语言到用C++定义的对象的链接是由实现定义的,并且依赖于语言。只有当两种语言实现的对象布局策略足够相似时,才能实现这种链接
这个答案是针对那些不耐烦/有最后期限的人的,下面只是部分/简单的解释:
在C++中,您可以通过重载在类中使用相同的名称(例如,由于它们都是相同的名称,因此无法从dll等导出)。解决这些问题的方法是将它们转换为不同的字符串(称为符号),符号说明了函数的名称,也说明了参数,因此即使这些函数具有相同的名称也可以唯一标识(也称为名称损坏)在C中,您没有重载,函数名是唯一的(因此,不需要单独的字符串来唯一标识函数名,因此符号是函数名本身)
所以在C++中,名称mangling唯一标识每个函数在C语言中,即使没有名字,也可以唯一地标识每个函数
要更改C++的行为,即指定不应对特定函数进行名称篡改,可以在函数名称之前使用外部“C”,无论出于何种原因,例如从dll导出具有特定名称的函数,供其客户端使用。
阅读其他答案,了解更详细/更正确的答案。
请参阅下面的链接,该链接是Geeks for Geeks对外部“C”用法的解释。从下面的页面添加重要信息。
考虑函数f()的以下声明
int f (void) { return 1; }
int f (int) { return 0; }
void g (void) { int i = f(), j = f(0); }
C++编译器可能会将上述名称更改为以下名称(来源:Wiki)
int __f_v (void) { return 1; }
int __f_i (int) { return 0; }
void __g_v (void) { int i = __f_v(), j = __f_i(0); }
https://www.geeksforgeeks.org/extern-c-in-c/
当混合使用C和C++时(即a.从C++调用C函数;b.从C调用C++函数),C++名称混乱会导致链接问题。从技术上讲,只有当被调用函数已经使用相应的编译器编译成二进制(很可能是*.a库文件)时,才会出现此问题。
因此,我们需要使用外部“C”来禁用C++中的名称篡改。
我只是想添加一些信息,因为我还没有看到它发布。
您将经常看到C头中的代码,如下所示:
#ifdef __cplusplus
extern "C" {
#endif
// all of your legacy C code here
#ifdef __cplusplus
}
#endif
这实现的是,它允许您在C++代码中使用C头文件,因为宏__cplusplus将被定义。但是您也可以将它与遗留的C代码一起使用,因为宏没有定义,所以它不会看到唯一的C++构造。
虽然,我也见过C++代码,例如:
extern "C" {
#include "legacy_C_header.h"
}
我想这也能完成同样的事情。
不知道哪种方式更好,但我都见过。