這樣的代碼到底是什麼意思呢?首先,__cplusplus是cpp中的自定義宏,那麼定義了這個宏的話表示這是一段cpp的代碼,也就是說,上面的代碼的含義是:如果這是一段cpp的代碼,那麼加入extern "C"{和}處理其中的代碼。
- #ifdef __cplusplus
- extern "C" {
- #endif
- //一段代碼
- #ifdef __cplusplus
- }
- #endif
要明白為何使用extern "C++",還得從cpp中對函數的重載處理開始說起。在C++中,為了支持重載機制,在編譯生成的匯編碼中,要對函數的名字進行一些處理,加入比如函數的返回類型等等.而在C++中,只是簡單的函數名字而已,不會加入其他的信息.也就是說:C++和C對產生的函數名字的處理是不一樣的.
比如下面的一段簡單的函數,我們看看加入和不加入extern "C++"產生的匯編代碼都有哪些變化:
- int f(void)
- {
- return 1;
- }
在加入extern "C"的時候產生的匯編代碼是:
- .file "test.cxx"
- .text
- .align 2
- .globl _f
- .def _f; .scl 2; .type 32; .endef
- _f:
- pushl %ebp
- movl %esp, %ebp
- movl $1, %eax
- popl %ebp
- ret
兩段匯編代碼同樣都是使用gcc -S命令產生的,所有的地方都是一樣的,唯獨是產生的函數名,一個是_f,一個是__Z1fv。明白了加入與不加入extern "C"之後對函數名稱產生的影響,我們繼續我們的討論:為什麼需要使用extern "C"呢?
C++之父在設計C++之時,考慮到當時已經存在了大量的C++代碼,為了支持原來的C++代碼和已經寫好C++庫,需要在C++中盡可能的支持C++,而extern "C++"就是其中的一個策略。試想這樣的情況:一個庫文件已經用C++寫好了而且運行得很良好。
這個時候我們需要使用這個庫文件,但是我們需要使用C++來寫這個新的代碼。如果這個代碼使用的是C++的方式鏈接這個C庫文件的話,那麼就會出現鏈接錯誤.我們來看一段代碼。
明白了加入與不加入extern "C++"之後對函數名稱產生的影響,我們繼續我們的討論:為什麼需要使用extern "C++"呢?C++之父在設計C++之時,考慮到當時已經存在了大量的C代碼,為了支持原來的C++代碼和已經寫好C++庫,需要在C++中盡可能的支持C++,而extern "C++"就是其中的一個策略。
試想這樣的情況:一個庫文件已經用C++寫好了而且運行得很良好,這個時候我們需要使用這個庫文件,但是我們需要使用C++來寫這個新的代碼。如果這個代碼使用的是C++的方式鏈接這個C++庫文件的話,那麼就會出現鏈接錯誤.我們來看一段代碼:首先,我們使用C的處理方式來寫一個函數。