extern C的作用详解

转载 2017年01月03日 19:50:52
 extern "C"的主要作用就是为了能够正确实现C++代码调用其他C语言代码。加上extern "C"后,会指示编译器这部分代码按C语言的进行编译,而不是C++的。由于C++支持函数重载,因此编译器编译函数的过程中会将函数的参数类型也加到编译后的代码中,而不仅仅是函数名;而C语言并不支持函数重载,因此编译C语言代码的函数时不会带上函数的参数类型,一般之包括函数名。


     这个功能十分有用处,因为在C++出现以前,很多代码都是C语言写的,而且很底层的库也是C语言写的,为了更好的支持原来的C代码和已经写好的C语言库,需要在C++中尽可能的支持C,而extern "C"就是其中的一个策略。


这个功能主要用在下面的情况:


1、C++代码调用C语言代码


2、在C++的头文件中使用


3、在多个人协同开发时,可能有的人比较擅长C语言,而有的人擅长C++,这样的情况下也会有用到


给出一个我设计的例子:


moduleA、moduleB两个模块,B调用A中的代码,其中A是用C语言实现的,而B是利用C++实现的,下面给出一种实现方法:


//moduleA头文件


#ifndef __MODULE_A_H //对于模块A来说,这个宏是为了防止头文件的重复引用


#define __MODULE_A_H


int fun(int, int);


#endif


//moduleA实现文件moduleA.C //模块A的实现部分并没有改变


#include"moduleA"


int fun(int a, int b)


{


return a+b;


}


//moduleB头文件


#idndef __MODULE_B_H //很明显这一部分也是为了防止重复引用


#define __MODULE_B_H


#ifdef __cplusplus //而这一部分就是告诉编译器,如果定义了__cplusplus(即如果是cpp文件, extern "C"{ //因为cpp文件默认定义了该宏),则采用C语言方式进行编译


#include"moduleA.h"


#endif


… //其他代码


#ifdef __cplusplus


}


#endif


#endif


//moduleB实现文件 moduleB.cpp //B模块的实现也没有改变,只是头文件的设计变化了


#include"moduleB.h"


int main()


{


cout<<fun(2,3)<<endl;


}


下面是详细的介绍:


由于C、C++编译器对函数的编译处理是不完全相同的,尤其对于C++来说,支持函数的重载,编译后的函数一般是以函数名和形参类型来命名的。


例如函数void fun(int, int),编译后的可能是(不同编译器结果不同)_fun_int_int(不同编译器可能不同,但都采用了类似的机制,用函数名和参数类型来命名编译后的函数名);而C语言没有类似的重载机制,一般是利用函数名来指明编译后的函数名的,对应上面的函数可能会是_fun这样的名字。


看下面的一个面试题:为什么标准头文件都有类似的结构?


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


#define __INCvxWorksh


#ifdef __cplusplus             //告诉编译器,这部分代码按C语言的格式进行编译,而不是C++的


extern "C"{


#endif


/*…*/


#ifdef __cplusplus


}


#endif


#endif /*end of __INCvxWorksh*/


分析:


显然,头文件中编译宏"#ifndef __INCvxWorksh 、#define __INCvxWorksh、#endif"(即上面代码中的蓝色部分)的作用是为了防止该头文件被重复引用
那么
#ifdef __cplusplus (其中__cplusplus是cpp中自定义的一个宏!!!)


extern "C"{


#endif


#ifdef __cplusplus


}


#endif


的作用是什么呢?


extern "C"包含双重含义,从字面上可以知道,首先,被它修饰的目标是"extern"的;其次,被它修饰的目标代码是"C"的。


被extern "C"限定的函数或变量是extern类型的
extern是C/C++语言中表明函数和全局变量的作用范围的关键字,该关键字告诉编译器,其申明的函数和变量可以在本模块或其他模块中使用。


记住,下面的语句:


extern int a; 仅仅是一个变量的声明,其并不是在定义变量a,并未为a分配空间。变量a在所有模块中作为一种全局变量只能被定义一次,否则会出错。


通常来说,在模块的头文件中对本模块提供给其他模块引用的函数和全局变量以关键字extern生命。例如,如果模块B要引用模块A中定义的全局变量和函数时只需包含模块A的头文件即可。这样模块B中调用模块A中的函数时,在编译阶段,模块B虽然找不到该函数,但并不会报错;它会在链接阶段从模块A编译生成的目标代码中找到该函数。


extern对应的关键字是static,static表明变量或者函数只能在本模块中使用,因此,被static修饰的变量或者函数不可能被extern C修饰。


被extern "C"修饰的变量和函数是按照C语言方式进行编译和链接的:这点很重要!!!!
上面也提到过,由于C++支持函数重载,而C语言不支持,因此函数被C++编译后在符号库中的名字是与C语言不同的;C++编译后的函数需要加上参数的类型才能唯一标定重载后的函数,而加上extern "C"后,是为了向编译器指明这段代码按照C语言的方式进行编译


未加extern "C"声明时的链接方式:


//模块A头文件 moduleA.h


#idndef _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这样的符号!!!,显然这是不可能找到的,因为foo()函数被编译成了_foo的符号,因此会出现链接错误。


常见的做法可以参考下面的一个实现:


moduleA、moduleB两个模块,B调用A中的代码,其中A是用C语言实现的,而B是利用C++实现的,下面给出一种实现方法:


//moduleA头文件


#ifndef __MODULE_A_H //对于模块A来说,这个宏是为了防止头文件的重复引用


#define __MODULE_A_H


int fun(int, int);


#endif


//moduleA实现文件moduleA.C //模块A的实现部分并没有改变


#include"moduleA"


int fun(int a, int b)


{


return a+b;


}


//moduleB头文件


#idndef __MODULE_B_H //很明显这一部分也是为了防止重复引用


#define __MODULE_B_H


#ifdef __cplusplus //而这一部分就是告诉编译器,如果定义了__cplusplus(即如果是cpp文件, extern "C"{ //因为cpp文件默认定义了该宏),则采用C语言方式进行编译


#include"moduleA.h"


#endif


… //其他代码


#ifdef __cplusplus


}


#endif


#endif


//moduleB实现文件 moduleB.cpp //B模块的实现也没有改变,只是头文件的设计变化了


#include"moduleB.h"


int main()


{


cout<<fun(2,3)<<endl;


}


extern "C"的使用要点


1. 可以是单一语句


    extern "C" double sqrt(double);


2. 可以是复合语句, 相当于复合语句中的声明都加了extern "C"


    extern "C"


   {


      double sqrt(double);


      int min(int, int);


  }


3.可以包含头文件,相当于头文件中的声明都加了extern "C"


   extern "C"


  {


    #i nclude <cmath>


  }


4. 不可以将extern "C" 添加在函数内部


5. 如果函数有多个声明,可以都加extern "C", 也可以只出现在第一次声明中,后面的声明会接受第一个链接指示符的规则。


6. 除extern "C", 还有extern "FORTRAN" 等。

extern C作用总结

extern C使用主要用与在C++代码中调用的C函数的声明,或C++中编译的函数要在C中调用,也即是导入C形式的函数库或者提供C类型的库给C调用,...
  • wangjiaoyu250
  • wangjiaoyu250
  • 2015年01月17日 19:48
  • 1358

C语言中关键字extern的作用

 1 基本解释   extern可以置于变量或者函数前,以标示变量或者函数的定义在别的文件中,提示编译器遇到此变量和函数时在其他模块中寻找其定义。   另外,extern也可用来进行链接指定。2 问题...
  • xiaozhi_su
  • xiaozhi_su
  • 2009年05月09日 13:56
  • 13157

C语言正确使用extern关键字

转载请注明出处利用关键字extern,可以在一个文件中引用另一个文件中定义的变量或者函数,下面就结合具体的实例,分类说明一下。 一、引用同一个文件中的变量#includeint func();int...
  • xingjiarong
  • xingjiarong
  • 2015年08月14日 08:37
  • 25477

C语言 static和extern关键字 对变量的作用

上一讲介绍了static和extern对函数的作用,static用来定义一个内部函数,不允许其他文件访问;extern用来定义和声明一个外部函数,允许其他文件访问。static和extern对变量也有...
  • VNanyesheshou
  • VNanyesheshou
  • 2016年01月11日 13:27
  • 2364

extern “C”作用详解

extern "C"的主要作用就是为了能够正确实现C++代码调用其他C语言代码。加上extern "C"后,会指示编译器这部分代码按C语言的进行编译,而不是C++的。由于C++支持函数重载,因此编译器...
  • zhu339111520
  • zhu339111520
  • 2014年02月28日 02:29
  • 351

extern "C"作用详解

extern “C”的主要作用就是为了能够正确实现C++代码调用其他C语言代码。加上extern “C”后,会指示编译器这部分代码按C语言的进行编译,而不是C++的。由于C++支持函数重载,因此编译器...
  • u010177286
  • u010177286
  • 2015年04月28日 17:25
  • 323

C++学习笔记(初级篇)——作用域与“extern”关键字

作用域,顾名思义,就是指某个标识符在程序中的有效区域。 在C++中,按作用域的大小,一般可分为局部作用域和全局作用域。 1.局部作用域 局部作用域用大括号,即{},来定义,大括号括起来的范围就是一个局...
  • Dr_Neo
  • Dr_Neo
  • 2015年12月11日 19:17
  • 1170

#define EXTERN

#define EXTERN  表示  把所有的EXTERN都换成空格 这样就消除了
  • cs18347264741
  • cs18347264741
  • 2014年09月14日 20:54
  • 857

extern C的作用详解

extern "C"的主要作用就是为了能够正确实现C++代码调用其他C语言代码。加上extern "C"后,会指示编译器这部分代码按C语言的进行编译,而不是C++的。由于C++支持函数重载,因此编译器...
  • cheatingdeath
  • cheatingdeath
  • 2013年12月12日 13:34
  • 396

extern "C"的作用详解

extern "C"的主要作用就是为了能够正确实现C++代码调用其他C语言代码。加上extern "C"后,会指示编译器这部分代码按c语言的进行编译,而不是C++的。由于C++支持函数重载,因此编...
  • qq_35624156
  • qq_35624156
  • 2017年08月20日 16:39
  • 69
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:extern C的作用详解
举报原因:
原因补充:

(最多只允许输入30个字)