在編譯器只認識.c(.cpp))檔案,而不知道.h是何物的年代,那時的人們寫了很多的.c(.cpp)檔案,漸漸地,人們發現在很多.c(.cpp)檔案中的宣告語句就是相同的,但他們卻不得不乙個字乙個字地重複地將這些內容敲入每個.c(.cpp)檔案。但更為恐怖的是,當其中乙個宣告有變更時,就需要檢查所有的.c(.cpp)檔案。
於是人們將重複的部分提取出來,放在乙個新檔案裡,然後在需要的.c(.cpp)檔案中敲入#include ***x這樣的語句。這樣即使某個宣告發生了變更,也再不需要到處尋找與修改了。因為這個新檔案,經常被放在.c(.cpp)檔案的頭部,所以就給它起名叫做「標頭檔案」,副檔名是.h。
在我們語言的初學階段,往往我們的程式只有乙個.c的檔案或這很少的幾個,這時我們就很少遇到頭檔案組織這個頭疼的問題,隨著我們程式的增加,** 量到了幾千行甚至幾萬行,檔案數也越來越多。這時這些檔案的組織就成了乙個問題,其實說白了這些檔案的組織問題從理論上來說是軟體工程中的模組設計等等的問題。
標頭檔案的作用的簡短描述:
(1)通過標頭檔案來呼叫庫功能。在很多場合,源**不便(或不准)向使用者公布,只要向使用者提供標頭檔案和二進位制的庫即可。使用者只需要按照標頭檔案中的介面宣告來呼叫庫功能,而不必關心介面怎麼實現的。編譯器會從庫中提取相應的**。
(2)標頭檔案能加強型別安全檢查。如果某個介面被實現或被使用時,其方式與標頭檔案中的宣告不一致,編譯器就會指出錯誤,這一簡單的規則能大大減輕程式設計師除錯、改錯的負擔。
比方說 我在aaa.h裡定義了乙個函式的宣告,然後我在aaa.h的同乙個目錄下建立aaa.c , aaa.c裡定義了這個函式的實現,然後是在main函式所在.c檔案裡#include這個aaa.h 然後我就可以使用這個函式了。 main在執行時就會找到這個定義了這個函式的aaa.c檔案。這是因為:main函式為標準c/c++的程式入口,編譯器會先找到該函式所在的檔案。假定編譯程式編譯myproj.c(其中含main())時,發現它include了mylib.h(其中宣告了函式void test()),那麼此時編譯器將按照事先設定的路徑(include路徑列表及**檔案所在的路徑)查詢與之同名的實現檔案(擴充套件名為.cpp或.c,此例中為mylib.c),如果找到該檔案,並在其中找到該函式(此例中為void test())的實現**,則繼續編譯;如果在指定目錄找不到實現檔案,或者在該檔案及後續的各include檔案中未找到實現**,則返回乙個編譯錯誤.其實include的過程完全可以「看成」是乙個檔案拼接的過程,將宣告和實現分別寫在標頭檔案及c檔案中,或者將二者同時寫在標頭檔案中,理論上沒有本質的區別。
理論上來說c檔案與標頭檔案裡的內容,只要是c語言所支援的,無論寫什麼都可以的,比如你在標頭檔案中寫函式體,只要在任何乙個c檔案包含此標頭檔案就可以將這個函式編譯成目標檔案的一部分(編譯是以c檔案為單位的,如果不在任何c檔案中包含此標頭檔案的話,這段**就形同虛設),你可以在c檔案中進行函式宣告,變數宣告,結構體宣告,這也不成問題!!!那為何一定要分成標頭檔案與c檔案呢?又為何一般都在頭件中進行函式,變數宣告,巨集宣告,結構體宣告呢?而在c檔案中去進行變數定義,函式實現呢??
要理解c檔案與標頭檔案有什麼不同之處,首先需要弄明白編譯器的工作過程,一般說來編譯器會做以下幾個過程:
1.預處理階段
2.詞法與語法分析階段
3.編譯階段,首先編譯成純彙編語句,再將之彙編成跟cpu相關的二進位製碼,生成各個目標檔案
4.連線階段,將各個目標檔案中的各段**進行絕對位址定位,生成跟特定平台相關的可執行檔案,編譯器在編譯時是以c檔案為單位進行的,也就是說如果你的專案中乙個c檔案都沒有,那麼你的專案將無法編譯,聯結器是以目標檔案為單位,它將乙個或多個目標檔案進行函式與變數的重定位,生成最終的可執行檔案,在pc上的程式開發,一般都有乙個main函式,這是各個編譯器的約定。為了生成乙個最終的可執行檔案,就需要一些目標檔案,也就是需要c檔案,而這些c檔案中又需要乙個main函式作為可執行程式的入口。
簡單些說就是c語言的編譯分為預處理、編譯、彙編、鏈結(test.c test.h => test.i => test.s => test.o => test)四個大的階段。c檔案中的#include巨集處理,會在預處理的階段將c中引用的**件的內容全部寫到c檔案中,最後生成.i中間檔案,這時h 檔案中的內容就相當於被寫道c檔案中。這也為**的復用提供了渠道,很多的c檔案可以去引用同乙個**件,這樣這個**件就會被放到多個c檔案中被編譯多 次,這也是**件中不能放定義只能放宣告的原因,放定義時被編譯多次,在程式鏈結的時候(系統中定義了多個int a;強符號定義)會出現錯誤, 宣告就不一樣,宣告表示對定義的擴充套件,最終都會終結到乙個定義上,所以不會出現link時重複定義的錯誤。
程式設計中我們在**件中肯定都用過一下的格式
#ifndef ***_h
#define ***_h
//……
#endif
呵呵,那他到底有什麼用呢,在**件互相引用時,消除重複定義。當然巨集定義是在預處理階段發揮作用的,編譯方後的過程是沒有巨集的影子的。
a.h
int a();
b.h#include "a.h"
c.h#include "a.h"
d.h#include "a.h"
#include "b.h"
上面的d.**件中就會重複出現兩個int a();的宣告阿,這樣就有點重複了,這時條件編譯巨集就派上了用場
a.h
#ifndef a_h
#define a_h
int a();
#endif
這樣就不會重複定義了。 c語言中的 c檔案和 h檔案
大神的詳細解讀 傳送門 記錄下一點點自己對兩個檔案的理解 函式宣告可以有多分,但函式定義只能有乙份。所以一般不在標頭檔案裡面定義函式,因為同乙個程式的多個 檔案可能都會包含這個標頭檔案。但c 中的inline函式是個例外,得定義在標頭檔案中。為了能夠擴充套件inline函式的內容,在每個呼叫點上,編...
C語言中C檔案與h標頭檔案的關係
簡單的說其實要理解c檔案與標頭檔案 即.h 有什麼不同之處,首先需要弄明白編譯器的工作過程,一般說來編譯器會做以下幾個過程 1.預處理階段 2.詞法與語法分析階段 3.編譯階段,首先編譯成純彙編語句,再將之彙編成跟cpu相關的二進位製碼,生成各個目標檔案 obj檔案 4.連線階段,將各個目標檔案中的...
c語言的 c和 h的檔案關係
c和.h檔案沒有必然聯絡,不一定非得成對存在,可單獨實現 h標頭檔案存在的意義,方便c檔案的引用 避免重複 與描述。理論上來說c檔案與標頭檔案裡的內容,只要是c語言所支援的,無論寫什麼都可以的,比如你在標頭檔案中寫函式體,只要在任何乙個c檔案包含此標頭檔案就可以將這個函式編譯成目標檔案的一部分 編譯...