一、extern
extern,是计算机语言中的一个关键字,可置于变量或者函数前,以表示变量或者函数的定义在别的文件中。提示编译器遇到此变量或函数时,在其它模块中寻找其定义,另外,extern也可用来进行链接指定。
声明可以多次,但是定义只能有一次。
(1) 变量
extern int a; //声明一个全局变量
int a; //定义一个全局变量
extern int a = 0;//定义全局变量并给初值
int a = 0; //定义全局变量并给初值
当要引用一个全局变量时,就要声明extern int a,这个时候extern不能省,否则就成定义了。
(2) 函数
函数也有声明和定义,但由于函数的声明和定义是有区别的,函数的定义是有函数体的,所以函数的声明和定义都可以将extern省略掉,反正其他文件也是知道这个函数是在其他地方定义的。
二、extern "C"
extern "C"的主要作用就是为了能够正确实现C++代码调用其他C语言代码。加上extern "C"后,会指示编译器这部分代码按C语言(而不是C++)的方式进行编译。由于C++支持函数重载,因此编译器编译函数的过程中会将函数的参数类型也加到编译后的代码中,而不仅仅是函数名;而C语言并不支持函数重载,因此编译C语言代码的函数时不会带上函数的参数类型,一般只包括函数名。
例如函数void fun(int, int),编译后的可能是_fun_int_int(不同编译器可能不同,但都采用了类似的机制,用函数名和参数类型来命名编译后的函数名);而C语言没有类似的重载机制,一般是利用函数名来指明编译后的函数名的,对应上面的函数可能会是_fun这样的名字。
举个例子:
test.h
#ifndef _HELLO_H_
#define _HELLO_H_
/* C++需加上extern "C" 声明 */
#ifdef __cplusplus
extern "C" {
#endif
void hello(int number);
#ifdef __cplusplus
}
#endif
#endif
test.cpp
#include "test.h"
#include <iostream>
void hello(int number){
std::cout<<"hello: "<<number<<std::endl;
}
int main()
{
int num = 666;
hello(num);
return 0;
}
编译运行:
下面一道面试题,为什么标准头文件都有类似的结构?
#ifndef __INCvxWorksh //防止该头文件被重复引用
#define __INCvxWorksh
#ifdef __cplusplus //告诉编译器,这部分代码按C语言的格式进行编译,而不是C++的
extern "C"{
#endif
/*…*/
#ifdef __cplusplus
}
#endif
#endif /*end of __INCvxWorksh*/
参考
[1] extern “C”的作用详解