C++extern “c”用法解析

应用场面

  • C++代码调用C语言代码、在C++的头文件中接纳
    在C++中援引C语言中的函数和变量,在蕴藏C语言头文件(固然为cExample.h)时,需举办下列处理:

extern "C"
{
#include "cExample.h"
}

而在C语言的头文件中,对其外表函数只可以指定为extern类型,C语言中不补助extern
“C”阐明,在.c文件中蕴含了extern “C”时会出现编译语法错误。

/* c语言头文件:cExample.h */
#ifndef C_EXAMPLE_H
#define C_EXAMPLE_H
extern int add(int x,int y);     //注:写成extern "C" int add(int , int ); 也可以
#endif

/* c语言实现文件:cExample.c */
#include "cExample.h"
int add( int x, int y )
{
 return x + y;
}

// c++实现文件,调用add:cppFile.cpp
extern "C"
{
 #include "cExample.h"        //注:此处不妥,如果这样编译通不过,换成 extern "C" int add(int , int ); 可以通过
}

int main(int argc, char* argv[])
{
 add(2,3);
 return 0;
}

倘使C++调用一个C语言编写的.DLL时,当包括.DLL的头文件或宣称接口函数时,应加extern
“C”{}。

  • 在C中援引C++语言中的函数和变量时,C++的头文件需添加extern
    “C”,可是在C语言中无法一向引用表明了extern
    “C”的该头文件,应该仅将C文件中校C++中定义的extern
    “C”函数表明为extern类型

//C++头文件 cppExample.h
#ifndef CPP_EXAMPLE_H
#define CPP_EXAMPLE_H
extern "C" int add( int x, int y );
#endif

//C++实现文件 cppExample.cpp
#include "cppExample.h"
int add( int x, int y )
{
 return x + y;
}

/* C实现文件 cFile.c
/* 这样会编译出错:#include "cExample.h" */
extern int add( int x, int y );
int main( int argc, char* argv[] )
{
 add( 2, 3 );
 return 0;
}

转载请表明作者杰森 Ding及其出处
Github主页(http://jasonding1354.github.io/)
CSDN博客(http://blog.csdn.net/jasonding1354)
简书主页(http://www.jianshu.com/users/2bd9b48f6ea8/latest\_articles)

从标准头文件说起

#ifndef __INCvxWorksh  /*防止该头文件被重复引用*/
#define __INCvxWorksh

#ifdef __cplusplus    //__cplusplus是cpp中自定义的一个宏
extern "C" {          //告诉编译器,这部分代码按C语言的格式进行编译,而不是C++的
#endif

    /**** some declaration or so *****/  

#ifdef __cplusplus
}
#endif

#endif /* __INCvxWorksh */

extern “C”的含义

extern “C”
包含重复意义,从字面上即可得到:首先,被它修饰的靶子是“extern”的;其次,被它修饰的目的是“C”的。
被extern “C”限定的函数或变量是extern类型的;
1、extern关键字
extern是C/C++语言中标明函数和全局变量功能范围(可见性)的首要字,该重大字告诉编译器,其申明的函数和变量可以在本模块或其他模块中采纳。
平常,在模块的头文件中对本模块提供给其余模块引用的函数和全局变量以第一字extern讲明。例如,即使模块B欲引用该模块A中定义的全局变量和函数时只需蕴涵模块A的头文件即可。这样,模块B中调用模块A中的函数时,在编译阶段,模块B尽管找不到该函数,可是并不会报错;它会在链接阶段中从模块A编译生成的对象代码中找到此函数。
与extern对应的第一字是static,被它修饰的全局变量和函数只好在本模块中接纳。因而,一个函数或变量只可能被本模块使用时,其不容许被extern
“C”修饰。

2、被extern “C”修饰的变量和函数是听从C语言格局编译和链接的
率先看望C++中对类似C的函数是何等编译的。
用作一种面向对象的言语,C++协理函数重载,而过程式语言C则不帮忙。函数被C++编译后在符号库中的名字与C语言的不等。例如,假如某个函数的原型为:
void foo( int x, int y );
该函数被C编译器编译后在符号库中的名字为_foo,而C++编译器则会生出像_foo_int_int之类的名字(不同的编译器可能生成的名字不同,不过都选拔了一如既往的建制,生成的新名字叫做“mangled
name”)。
**
_foo_int_int这样的名字包含了函数名、函数参数数量及类型信息,C++就是靠这种体制来贯彻函数重载的。**
例如,在C++中,函数void foo( int x, int y )与void foo( int x, float y
)编译生成的符号是不均等的,后者为_foo_int_float。
一样地,C++中的变量除援助部分变量外,还襄助类成员变量和全局变量。用户所编写程序的类成员变量可能与全局变量同名,我们以”.”来分别。而实质上,编译器在举行编译时,与函数的拍卖一般,也为类中的变量取了一个旷世的名字,那些名字与用户程序中同名的全局变量名字不同。

3、举例表明
(1)未加extern “C”阐明时的连天情势
假如在C++中,模块A的头文件如下:

// 模块A头文件 moduleA.h
#ifndef MODULE_A_H
#define MODULE_A_H
int foo( int x, int y );
#endif

//在模块B中引用该函数:
// 模块B实现文件 moduleB.cpp
#include "moduleA.h"
foo(2,3);

实在,在接连阶段,链接器会从模块A生成的对象文件moduleA.obj中查找_foo_int_int这样的标记!

(2)加extern “C”表明后的编译和链接情势
加extern “C”表明后,模块A的头文件变为:

// 模块A头文件 moduleA.h
#ifndef MODULE_A_H
#define MODULE_A_H
extern "C" int foo( int x, int y );
#endif

在模块B的落实公文中仍然调用foo( 2,3 ),其结果是:
<1>A编译生成foo的对象代码时,没有对其名字进行超常规处理,采纳了C语言的格局;
<2>链接器在为模块B的目标代码寻找foo(2,3)调用时,寻找的是未经修改的记号名_foo。

万一在模块A中函数注脚了foo为extern “C”类型,而模块B中带有的是extern int
foo(int x, int y),则模块B找不到模块A中的函数;反之亦然。

extern “C”这多少个宣称的忠实目标是为了实现C++与C及其他语言的混合编程

引言

C++保留了一片段过程式语言的表征,因此它可以定义不属于其他类的全局变量和函数。可是,C++毕竟是一种面向对象的程序设计语言,为了帮助函数的重载,C++对全局函数的处理格局与C有分明的例外。
extern
“C”的首要功用就是为着可以科学贯彻C++代码调用其他C语言代码。加上extern
“C”后,会指示编译器这部分代码按C语言的展开编译,而不是C++的。由于C++帮忙函数重载,由此编译器编译函数的过程中会将函数的参数类型也加到编译后的代码中,而不仅是函数名;而C语言并不协助函数重载,因而编译C语言代码的函数时不会带上函数的参数类型,一般之包括函数名。

譬如说你用C 开发了一个DLL 库,为了可以让C
++语言也可以调用你的DLL输出(Export)的函数,你需要用extern
“C”来强制编译器不要改动你的函数名。

揭秘extern “C”