Linux環境下GNU, GCC, G++編譯器(轉)
一,GNU
GNU是“GNU ‘s Not Unix”的遞歸縮寫, Stallman宣布GNU應當發音為Guh-NOO(革奴)以避免與new這個單詞混淆(註:Gnu在英文中原意為非洲牛羚,發音與new相同)
為保證GNU軟件可以自由地“使用、復制、修改和發布”,所有GNU軟件都在一份在禁止其他人添加任何限制的情況下授權所有權利給任何人的協議條款,GNU通用公共許可證(GNU General Public License,GPL)。這個就是被稱為“反版權”(或稱Copyleft)的概念。
GUN可以理解為一種linux規範。
二,gcc與g++的區別
gcc和g++都是GNU(組織)的一個編譯器。對它們的認識有很多誤區:
【誤區一】gcc只能編譯c代碼,g++只能編譯c++代碼
兩者都可以,但是請註意:
1.後綴為.c的,gcc把它當作是C程序,而g++當作是c++程序;後綴為.cpp的,兩者都會認為是c++程序,註意,雖然c++是c的超集,但是兩者對語法的要求是有區別的,例如:
1 #include 2 3 int main(int argc, char* argv[]) 4 5 { 6 if(argv == 0) return; 7 printString(argv); 8 return; 9 10 } 11 12 int printString(char* string) 13 14 { 15 sprintf(string, "This is a test.\n"); 16 17 }
如果按照C的語法規則,OK,沒問題,但是,一旦把後綴改為cpp,立刻報三個錯:“printString未定義”;
“cannot convert `char**‘ to `char*”;
”return-statement with no value“;
分別對應前面紅色標註的部分。可見C++的語法規則更加嚴謹一些。
2.編譯階段,g++會調用gcc,對於c++代碼,兩者是等價的,但是因為gcc命令不能自動和C++程序使用的庫聯接,所以通常用g++來完成鏈接,為了統一起見,幹脆編譯/鏈接統統用g++了,這就給人一種錯覺,好像cpp程序只能用g++似的。
【誤區二】:gcc不會定義__cplusplus宏,而g++會
實際上,這個宏只是標誌著編譯器將會把代碼按C還是C++語法來解釋,如上所述,如果後綴為.c,並且采用gcc編譯器,則該宏就是未定義的,否則,就是已定義。
【誤區三】:編譯只能用gcc,鏈接只能用g++
嚴格來說,這句話不算錯誤,但是它混淆了概念,應該這樣說:編譯可以用gcc/g++,而鏈接可以用g++或者gcc -lstdc++。因為gcc命令不能自動和C++程序使用的庫聯接,所以通常使用g++來完成聯接。但在編譯階段,g++會自動調用gcc,二者等價。
【誤區四】:extern "C"與gcc/g++有關系
實際上並無關系,無論是gcc還是g++,用extern "c"時,都是以C的命名方式來為symbol命名,否則,都以c++方式命名。試驗如下:
1 me.h: 2 extern "C" void CppPrintf(void); 3 4 me.cpp: 5 #include 6 #include "me.h" 7 using namespace std; 8 void CppPrintf(void) 9 { 10 cout 11 #include 12 #include "me.h" 13 int main(void) 14 { 15 CppPrintf(); 16 return 0; 17 }
1. 先給me.h加上extern "C",看用gcc和g++命名有什麽不同
[[email protected] G++]# g++ -S me.cpp
[[email protected] G++]# less me.s
.globl _Z9CppPrintfv //註意此函數的命名
.type CppPrintf, @function
[[email protected] GCC]# gcc -S me.cpp
[[email protected] GCC]# less me.s
.globl _Z9CppPrintfv //註意此函數的命名
.type CppPrintf, @function
完全相同!
2. 去掉me.h中extern "C",看用gcc和g++命名有什麽不同
[[email protected] GCC]# gcc -S me.cpp
[[email protected] GCC]# less me.s
.globl _Z9CppPrintfv //註意此函數的命名
.type _Z9CppPrintfv, @function
[[email protected] G++]# g++ -S me.cpp
[[email protected] G++]# less me.s
.globl _Z9CppPrintfv //註意此函數的命名
.type _Z9CppPrintfv, @function
完全相同!
可見extern "C"與采用gcc/g++並無關系,以上的試驗還間接的印證了前面的說法:在編譯階段,g++是調用gcc的。
轉自:http://www.linuxidc.com/Linux/2014-07/104299.htm
Linux環境下GNU, GCC, G++編譯器(轉)