结论:
1.gcc认为.c的为C程序,.cpp的为C++程序;
2.g++认为.c的为C++程序,.cpp的为C++程序;
3.VC++的编译器cl认为.c的为C程序,.cpp的为C++程序;
4.C程序与C++程序中同样的函数在编译后的obj文件中的symbol是不同的,所以以C方式编译的obj文件与以C++方式编译的obj文件无法成功链接。
使个demo说明一下:
准备工作:
为gcc、g++、cl(VC++编译器)、link(VC++链接器)设置好环境变量
//demo.cpp
#include
#include "foo.h"
int main()
{
printHello();
return 0;
}
//foo.h
void printHello();
//foo.c
#include
void printHello()
{
printf("Hello MM");
}
1.gcc、g++测试,在windows的cmd下:
D:\>g++ demo.cpp -o demo.obj -c (以C++方式编译生成demo.obj文件,-c选项表示只编译不链接)
D:\>gcc foo.c -o foo.obj -c (以C方式编译生成foo.obj文件)
D:\>g++ demo.obj foo.obj -o demo (链接demo.obj、foo.obj文件)
demo.obj(.text+0x2b):demo.cpp: undefined reference to `printHello()'
collect2: ld returned 1 exit status
提示说找不到printHello,因为按照C++的编译方式去找printHello应该对应某一种格式的symbol,但是我们
的foo.obj中printHello的symbol是另外一种格式的,所以找不到了。
把foo.c改成foo.cpp就可以成功链接了。
2.cl、link测试,在windows的cmd下:
D:\>cl demo.cpp /c (以C++方式编译生成demo.obj文件,/c选项表示只编译不链接)
D:\>cl foo.c /c (以C方式编译生成foo.obj文件)
D:\>link demo.obj foo.obj
Microsoft (R) Incremental Linker Version 6.00.8168
Copyright (C) Microsoft Corp 1992-1998. All rights reserved.
demo.obj : error LNK2001: unresolved external symbol "void __cdecl printHello(void)"(?printHello@@YAXXZ)
demo.exe : fatal error LNK1120: 1 unresolved externals
理由同上,把foo.c改成foo.cpp就可以成功链接了。
参考文章:
1.gcc和g++的区别(开源CEO)
http://www.linuxdiyf.com/bbs/viewthread.php?tid=109684
2.在 console mode 中使用 C/C++ 编译器(jjhou候俊杰)
http://jjhou.csdn.net/article99-10.htm
没有整理与归纳的知识,一文不值!高度概括与梳理的知识,才是自己真正的知识与技能。 永远不要让自己的自由、好奇、充满创造力的想法被现实的框架所束缚,让创造力自由成长吧! 多花时间,关心他(她)人,正如别人所关心你的。理想的腾飞与实现,没有别人的支持与帮助,是万万不能的。

本文详细探讨了C与CPP后缀文件在不同编译器下的处理方式,包括gcc/g++和VC++编译器。实验表明,文件后缀直接影响编译器如何解析函数符号,导致链接阶段可能出现错误。理解这种差异对于解决编译链接问题至关重要。

被折叠的 条评论
为什么被折叠?



