参考:
- http://blog.csdn.net/nana08/article/details/6910294
- http://www.52pojie.cn/thread-58109-1-1.html
- http://book.2cto.com/201306/25348.html
gcc和g++都是GNU的编译器,区别
- 对于.c后缀的文件,gcc把它当做是C程序;g++当做是C++程序
- 对于.cpp后缀的文件,gcc和g++都会当做C++程序
- 编译阶段,g++会调用gcc
- 连接阶段,通常会用g++来完成,这是因为gcc命令不能自动和C++程序使用的库链接,可以使用gcc -lstdc++
- 无论是gcc还是g++,用extern "c"时,都是以C的命名方式来为symbol命名,否则,都以c++方式命名
gcc/g++在执行编译工作的时候,总共需要4步
- 预处理,生成.i的文件[预处理器cpp]
- 将预处理后的文件转换成汇编语言,生成文件.s[编译器egcs]
- 有汇编变为目标代码(机器代码)生成.o的文件[汇编器as]
- 连接目标代码,生成可执行程序[链接器ld]
宏__cplusplus
#ifdef __cplusplus
extern "C" {
#endif
// 一些代码
#ifdef __cplusplus
}
#endif
这种类型的头文件可以被#include到C文件中进行编译,也可以被#include到C++文件中进行编译。由于extern "C"可以抑制C++对函数名、变量名等符号(symbol)进行名称重整(name mangling),因此编译出的C目标文件和C++目标文件中的变量、函数名称等符号都是相同的(否则不相同),链接器可以可靠地对两种类型的目标文件进行链接。这样该做法成为了C与C++混用头文件的典型做法。
鉴于以上的做法,程序员可能认为__cplusplus这个宏只有“被定义了”和“未定义”两种状态。事实上却并非如此,__cplusplus这个宏通常被定义为一个整型值。而且随着标准变化,__cplusplus宏一般会是一个比以往标准中更大的值。比如在C++03标准中,__cplusplus的值被预定为199711L,而在C++11标准中,宏__cplusplus被预定义为201103L。这点变化可以为代码所用。比如程序员在想确定代码是使用支持C++11编译器进行编译时,那么可以按下面的方法进行检测:
#if __cplusplus < 201103L
#error "should use C++11 implementation"
#endif
这里,使用了预处理指令#error,这使得不支持C++11的代码编译立即报错并终止编译。读者可以使用C++98编译器和C++11的编译器分别实验一下其效果。