gcc,g++,extern “C” :一些编译错误的缘由

正好是我们代码中遇到的问题,之前不求甚解,只用g++编译没有错误就不管了,现在要跨平台到windows下就遇到了问题。
全文转载自:http://user.qzone.qq.com/75172588/blog/1248167335

gcc和g++的区别
1、gcc
在编译阶段,把后缀为.c的源文件按照c语法和方式进行编译;把后缀为.cpp的源文件,按照c++语法和方式进行编译。
在链接阶段,不自动和C++程序使用的库链接。所以,.cpp如果有c++的代码,可能会产生链接错误;.c不存在这个问题,因为如果.c如果有c++的代码,编译时就会不通过。
2、g++
在编译阶段,无论是.c还是.cpp,都按照c++的语法和方式进行编译。其实,g++是调用gcc进行编译。
在链接阶段,会自动和C++程序使用的库链接,即便是纯c程序,也会链接这些库。

为什么需要extern “C”?
在c++中,为了支持重载机制,在编译生成的汇编码中,要对函数的名字进行一些处理,加入比如函数的返回类型等等。而在C中,只是简单的函数名字而已,不会加入其他的信息。也就是说,C++和C对产生的函数名字的处理是不一样的。所以,在gcc和g++下编译,会有如下效果:
gcc编译.c文件,因为是按照c方式编译,所以函数名不变;
gcc编译.cpp文件,g++编译.c文件,g++编译.cpp文件,因为是按照c++方式编译,所以函数名加上了附加信息。

在被编译的c++程序中,如果调用了其他的库的函数,则这个函数名也是照加了附加信息的函数名来调的,这样,如果这个库是c方式编译的,那么在库中的函数名是不带附加信息的,这样的调用链接就会失败,因为函数名不对应导致找不到。

在这样的场景下,就需要引入extern “C”。
extern “C”是告诉编译器:这是一个用C写成的库文件,请用C的方式来链接它们。
一般的用法是,比如,现在我们有了一个C库文件,它的头文件是f.h,产生的lib文件是f.lib,那么我们如果要在C++中使用这个库文件,我们需要这样写:

1

2

3

4

; html-script: false ]extern "C"

{

#include "f.h"

}

另外,被调函数的实现如果包在extern “C”中,意味着强制按c方式来编译,具体如下:
gcc编译.c文件,会编译不通过,因为c语法中没有extern “C”,实际上在这种情况下根本不需要把函数实现包在extern “C”中,因为这种情况下本来就是按c方式来编译的。因此通常会把代码写成这样:

1

2

3

4

5

6

7

8

9

; html-script: false ]#ifdef __cplusplus

extern "C" {

#endif

//一段代码

#ifdef __cplusplus

}

#endif

__cplusplus宏标志着编译器将会把代码按C还是C++语法来解释,如上所述,如果后缀为.c,并且采用gcc编译器,则该宏就是未定义的,否则,就是已定义的。
gcc编译.cpp文件,g++编译.c文件,g++编译.cpp文件,函数名不变,即采用c的方式。

为什么要把被调函数的实现包在extern “C”里呢?我想是为了使c程序也能调用它,否则就没有什么必要了。

后记:cl编译器的规则与gcc相同。

时间: 2024-10-07 20:06:10

gcc,g++,extern “C” :一些编译错误的缘由的相关文章

gcc/g++以c++11编译

方法一: //在程序头加上预定义编译器命令 #pragma GCC diagnostic error "-std=c++11" //通过#pragma 指示 GCC编译器处理错误的方式以c++11标准; 方法二: //在编译指令中加-std=c++11 g++ test.cpp -o test -std=c++11 //在给一个模板类定义别名,用using时,g++会报错 error: expected unqualified-id before 'using'    //就这个错误就

gcc新版本号引起的编译错误(命令运行时的外部库输入位置)

昨天,遇到一个比較bug的错误,用gcc来编译几个简单的文件出错,编译环境为x86_64的Ubuntu12.04.gcc版本号号例如以下: gcc (Ubuntu/Linaro 4.6.3-1ubuntu5) 4.6.3 编译例如以下命令,主要是须要加入一个经常使用数学库libm.so.编译的语法例如以下出现了例如以下的错误: gcc -lm main.c bp_alg_api.c -o test 还是会出现没有链接时指定的库文件lm. 假设这样就能够成功: gcc main.c bp_alg_

gcc g++支持C++11 标准编译及其区别

g++ -g -Wall -std=c++11 main.cpp gcc -g -Wall -std=c11 main.cpp 如果不想每次写这个-std=C++11这个选项该怎么办呢? 方法出处:http://stackoverflow.com/questions/16886591/how-do-i-enable-c11-in-gcc 方法1:写Makefile 方法2:取别名 :alias g++11="g++ -std=c++11" -----------------------

【转】gcc/g++ 如何支持c11 / c++11标准编译

如果用命令 g++ -g -Wall main.cpp  编译以下代码 : 1 2 3 4 5 6 7 8 9 10 11 12 /*     file : main.cpp */ #include <stdio.h> int main() {     int a[5] = { 1, 2, 2, 5, 1 };     for( int i:a ) {         printf( "%d\n", a[i] );     }     return 0; } 那么g++ 就

gcc/g++ 如何支持c11 / c++11标准编译

如果用命令 g++ -g -Wall main.cpp  编译以下代码 : /* file : main.cpp */ #include <stdio.h> int main() { int a[5] = { 1, 2, 2, 5, 1 }; for( int i:a ) { printf( "%d\n", a[i] ); } return 0; } 那么g++ 就会提示以下错误: main.cpp: In function ‘int main()’: main.cpp:5

gcc/g++等编译器 编译原理: 预处理,编译,汇编,链接各步骤详解

摘自http://blog.csdn.net/elfprincexu/article/details/45043971 gcc/g++等编译器 编译原理: 预处理,编译,汇编,链接各步骤详解 C和C++编译器是集成的,编译一般分为四个步骤: 预处理(preprocessing)  ----------------- cpp/ gcc -E  编译(compilation) ------------------ cc1 / gcc -S 汇编(assembly)  ----------------

gcc/g++ 实战之编译过程

gcc和g++分别是GNU(一个开源组织)的c&c++编译器 对于.c后缀的文件,gcc把它当做是C程序,g++当做是C++程序:对于.cpp后缀的文件,gcc和g++都会当做c++程序.gcc可以根据后缀名为.c或.cpp分别按c程序和c++程序来编译,但是g++无论是.c或.cpp都统一按c++程序来编译. 编译阶段,g++会调用gcc,链接阶段通常会用g++来完成,g++在编译的过程中,其实是调用gcc按照c++程序来编译的.即编译工作最终都是由gcc来完成的.这是因为gcc命令不能自动和

gcc/g++ 实战之编译的四个过程

gcc和g++分别是GNU(一个开源组织)的c&c++编译器 对于.c后缀的文件,gcc把它当做是C程序,g++当做是C++程序:对于.cpp后缀的文件,gcc和g++都会当做c++程序.gcc可以根据后缀名为.c或.cpp分别按c程序和c++程序来编译,但是g++无论是.c或.cpp都统一按c++程序来编译. 编译阶段,g++会调用gcc,链接阶段通常会用g++来完成,g++在编译的过程中,其实是调用gcc按照c++程序来编译的.即编译工作最终都是由gcc来完成的.这是因为gcc命令不能自动和

gcc/g++/make 编译信息带颜色输出

如果编译一个项目错误警告太多,非常不好找,所以非常希望输出信息可以带有颜色. 可是 gcc 4.9.0 之前的版本并不支持,很多情况下是不能替换编译器的,比如使用交叉编译器, 也可以使用 colorgcc,但我觉得不是特别好,需要配置,如果使用Makefile还要更改编译器设置, 所以我自己动手写了一个,效果还可以,源码在github上: https://github.com/chinaran/color-compile, 也可以去CSDN下载,点我. ---------------------