亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > C++ > 正文

C++中頭文件和源文件詳細介紹

2020-05-23 13:47:08
字體:
來源:轉載
供稿:網友

C++中的頭文件和源文件詳解

一、C++編譯模式

通常,在一個C++程序中,只包含兩類文件——.cpp文件和.h文件。其中,.cpp文件被稱作C++源文件,里面放的都是C++的源代碼;而.h文件則被稱作C++頭文件,里面放的也是C++的源代碼。

C+ +語言支持“分別編譯”(separate compilation)。也就是說,一個程序所有的內容,可以分成不同的部分分別放在不同的.cpp文件里。.cpp文件里的東西都是相對獨立的,在編 譯(compile)時不需要與其他文件互通,只需要在編譯成目標文件后再與其他的目標文件做一次鏈接(link)就行了。比如,在文件a.cpp中定義 了一個全局函數“void a() {}”,而在文件b.cpp中需要調用這個函數。即使這樣,文件a.cpp和文件b.cpp并不需要相互知道對方的存在,而是可以分別地對它們進行編譯, 編譯成目標文件之后再鏈接,整個程序就可以運行了。

這是怎么實現的呢?從寫程序的角度來講,很簡單。在文件b.cpp中,在調用 “void a()”函數之前,先聲明一下這個函數“void a();”,就可以了。這是因為編譯器在編譯b.cpp的時候會生成一個符號表(symbol table),像“void a()”這樣的看不到定義的符號,就會被存放在這個表中。再進行鏈接的時候,編譯器就會在別的目標文件中去尋找這個符號的定義。一旦找到了,程序也就可以 順利地生成了。

注意這里提到了兩個概念,一個是“定義”,一個是“聲明”。簡單地說,“定義”就是把一個符號完完整整地描述出來:它是變 量還是函數,返回什么類型,需要什么參數等等。而“聲明”則只是聲明這個符號的存在,即告訴編譯器,這個符號是在其他文件中定義的,我這里先用著,你鏈接 的時候再到別的地方去找找看它到底是什么吧。定義的時候要按C++語法完整地定義一個符號(變量或者函數),而聲明的時候就只需要寫出這個符號的原型了。 需要注意的是,一個符號,在整個程序中可以被聲明多次,但卻要且僅要被定義一次。試想,如果一個符號出現了兩種不同的定義,編譯器該聽誰的?

這 種機制給C++程序員們帶來了很多好處,同時也引出了一種編寫程序的方法??紤]一下,如果有一個很常用的函數“void f() {}”,在整個程序中的許多.cpp文件中都會被調用,那么,我們就只需要在一個文件中定義這個函數,而在其他的文件中聲明這個函數就可以了。一個函數還 好對付,聲明起來也就一句話。但是,如果函數多了,比如是一大堆的數學函數,有好幾百個,那怎么辦?能保證每個程序員都可以完完全全地把所有函數的形式都 準確地記下來并寫出來嗎?

二、什么是頭文件

很顯然,答案是不可能。但是有一個很簡單地辦法,可以幫助程序員們省去記住那么多函數原型的麻煩:我們可以把那幾百個函數的聲明語句全都先寫好,放在一個文件里,等到程序員需要它們的時候,就把這些東西全部copy進他的源代碼中。

這 個方法固然可行,但還是太麻煩,而且還顯得很笨拙。于是,頭文件便可以發揮它的作用了。所謂的頭文件,其實它的內容跟.cpp文件中的內容是一樣的,都是 C++的源代碼。但頭文件不用被編譯。我們把所有的函數聲明全部放進一個頭文件中,當某一個.cpp源文件需要它們時,它們就可以通過一個宏命令 “#include”包含進這個.cpp文件中,從而把它們的內容合并到.cpp文件中去。當.cpp文件被編譯時,這些被包含進去的.h文件的作用便發 揮了。

舉一個例子吧,假設所有的數學函數只有兩個:f1和f2,那么我們把它們的定義放在math.cpp里:

/* math.cpp */double f1(){ //do something here.... return;}double f2(double a){ //do something here... return a * a;}/* end of math.cpp */

并把“這些”函數的聲明放在一個頭文件math.h中:

/* math.h */double f1();double f2(double);/* end of math.h */

在另一個文件main.cpp中,我要調用這兩個函數,那么就只需要把頭文件包含進來:

/* main.cpp */#include "math.h"main(){ int number1 = f1(); int number2 = f2(number1);}/* end of main.cpp */

這 樣,便是一個完整的程序了。需要注意的是,.h文件不用寫在編譯器的命令之后,但它必須要在編譯器找得到的地方(比如跟main.cpp在一個目錄下)。 main.cpp和math.cpp都可以分別通過編譯,生成main.o和math.o,然后再把這兩個目標文件進行鏈接,程序就可以運行了。

三、#include

#include 是一個來自C語言的宏命令,它在編譯器進行編譯之前,即在預編譯的時候就會起作用。#include的作用是把它后面所寫的那個文件的內容,完完整整地、 一字不改地包含到當前的文件中來。值得一提的是,它本身是沒有其它任何作用與副功能的,它的作用就是把每一個它出現的地方,替換成它后面所寫的那個文件的 內容。簡單的文本替換,別無其他。因此,main.cpp文件中的第一句(#include "math.h"),在編譯之前就會被替換成math.h文件的內容。即在編譯過程將要開始的時候,main.cpp的內容已經發生了改變:

/* ~main.cpp */double f1();double f2(double);main(){ int number1 = f1(); int number2 = f2(number1);}/* end of ~main.cpp */

不多不少,剛剛好。同理可知,如果我們除了main.cpp以外,還有其他的很多.cpp文件也用到了f1和f2函數的話,那么它們也通通只需要在使用這兩個函數前寫上一句#include "math.h"就行了。

四、頭文件中應該寫什么

通 過上面的討論,我們可以了解到,頭文件的作用就是被其他的.cpp包含進去的。它們本身并不參與編譯,但實際上,它們的內容卻在多個.cpp文件中得到了 編譯。通過“定義只能有一次”的規則,我們很容易可以得出,頭文件中應該只放變量和函數的聲明,而不能放它們的定義。因為一個頭文件的內容實際上是會被引 入到多個不同的.cpp文件中的,并且它們都會被編譯。放聲明當然沒事,如果放了定義,那么也就相當于在多個文件中出現了對于一個符號(變量或函數)的定 義,縱然這些定義都是相同的,但對于編譯器來說,這樣做不合法。

所以,應該記住的一點就是,.h頭文件中,只能存在變量或者函數的聲明, 而不要放定義。即,只能在頭文件中寫形如:extern int a;和void f();的句子。這些才是聲明。如果寫上int a;或者void f() {}這樣的句子,那么一旦這個頭文件被兩個或兩個以上的.cpp文件包含的話,編譯器會立馬報錯。(關于extern,前面有討論過,這里不再討論定義跟 聲明的區別了。)但是,這個規則是有三個例外的。

一,頭文件中可以寫const對象的定義。因為全局的const對象默 認是沒有extern的聲明的,所以它只在當前文件中有效。把這樣的對象寫進頭文件中,即使它被包含到其他多個.cpp文件中,這個對象也都只在包含它的 那個文件中有效,對其他文件來說是不可見的,所以便不會導致多重定義。同時,因為這些.cpp文件中的該對象都是從一個頭文件中包含進去的,這樣也就保證 了這些.cpp文件中的這個const對象的值是相同的,可謂一舉兩得。同理,static對象的定義也可以放進頭文件。

二,頭文件中可 以寫內聯函數(inline)的定義。因為inline函數是需要編譯器在遇到它的地方根據它的定義把它內聯展開的,而并非是普通函數那樣可以先聲明再鏈 接的(內聯函數不會鏈接),所以編譯器就需要在編譯時看到內聯函數的完整定義才行。如果內聯函數像普通函數一樣只能定義一次的話,這事兒就難辦了。因為在 一個文件中還好,我可以把內聯函數的定義寫在最開始,這樣可以保證后面使用的時候都可以見到定義;但是,如果我在其他的文件中還使用到了這個函數那怎么辦 呢?這幾乎沒什么太好的解決辦法,因此C++規定,內聯函數可以在程序中定義多次,只要內聯函數在一個.cpp文件中只出現一次,并且在所有的.cpp文 件中,這個內聯函數的定義是一樣的,就能通過編譯。那么顯然,把內聯函數的定義放進一個頭文件中是非常明智的做法。

三,頭文件中可以寫類 (class)的定義。因為在程序中創建一個類的對象時,編譯器只有在這個類的定義完全可見的情況下,才能知道這個類的對象應該如何布局,所以,關于類的 定義的要求,跟內聯函數是基本一樣的。所以把類的定義放進頭文件,在使用到這個類的.cpp文件中去包含這個頭文件,是一個很好的做法。在這里,值得一提 的是,類的定義中包含著數據成員和函數成員。數據成員是要等到具體的對象被創建時才會被定義(分配空間),但函數成員卻是需要在一開始就被定義的,這也就 是我們通常所說的類的實現。一般,我們的做法是,把類的定義放在頭文件中,而把函數成員的實現代碼放在一個.cpp文件中。這是可以的,也是很好的辦法。 不過,還有另一種辦法。那就是直接把函數成員的實現代碼也寫進類定義里面。在C++的類中,如果函數成員在類的定義體中被定義,那么編譯器會視這個函數為 內聯的。因此,把函數成員的定義寫進類定義體,一起放進頭文件中,是合法的。注意一下,如果把函數成員的定義寫在類定義的頭文件中,而沒有寫進類定義中, 這是不合法的,因為這個函數成員此時就不是內聯的了。一旦頭文件被兩個或兩個以上的.cpp文件包含,這個函數成員就被重定義了。

五、頭文件中的保護措施

考 慮一下,如果頭文件中只包含聲明語句的話,它被同一個.cpp文件包含再多次都沒問題——因為聲明語句的出現是不受限制的。然而,上面討論到的頭文件中的 三個例外也是頭文件很常用的一個用處。那么,一旦一個頭文件中出現了上面三個例外中的任何一個,它再被一個.cpp包含多次的話,問題就大了。因為這三個 例外中的語法元素雖然“可以定義在多個源文件中”,但是“在一個源文件中只能出現一次”。設想一下,如果a.h中含有類A的定義,b.h中含有類B的定 義,由于類B的定義依賴了類A,所以b.h中也#include了a.h?,F在有一個源文件,它同時用到了類A和類B,于是程序員在這個源文件中既把 a.h包含進來了,也把b.h包含進來了。這時,問題就來了:類A的定義在這個源文件中出現了兩次!于是整個程序就不能通過編譯了。你也許會認為這是程序 員的失誤——他應該知道b.h包含了a.h——但事實上他不應該知道。

使用"#define"配合條件編譯可以很好地解決這個問題。在一 個頭文件中,通過#define定義一個名字,并且通過條件編譯#ifndef...#endif使得編譯器可以根據這個名字是否被定義,再決定要不要繼 續編譯該頭文中后續的內容。這個方法雖然簡單,但是寫頭文件時一定記得寫進去。

感謝閱讀,希望能幫助到大家,謝謝大家對本站的支持!


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
在线成人激情视频| 亚洲欧美中文日韩v在线观看| 国产精品久久久久免费a∨大胸| 韩剧1988在线观看免费完整版| 成人a级免费视频| 91国内揄拍国内精品对白| 国产午夜精品视频免费不卡69堂| 在线播放日韩欧美| 热久久免费视频精品| 欧美在线性视频| 国产精品久久久久久网站| 亚洲最大av网| 青青青国产精品一区二区| 国产成人亚洲综合青青| 亚洲福利视频专区| 亚洲人成网在线播放| 国产成人亚洲综合91| 92版电视剧仙鹤神针在线观看| 国产精品视频99| 亚洲成年网站在线观看| 国产精品日韩久久久久| www.亚洲男人天堂| 国产精品自产拍在线观| 91久久精品日日躁夜夜躁国产| 岛国视频午夜一区免费在线观看| 亚洲xxxxx电影| 久久久久久久一| 欧美高清在线播放| 日本精品一区二区三区在线| 欧美在线影院在线视频| 国产精品视频内| 色婷婷综合久久久久中文字幕1| 国产午夜精品久久久| 国产精品一区二区三| 国产欧美日韩高清| 色无极影院亚洲| 国内精品免费午夜毛片| 久久av红桃一区二区小说| 69精品小视频| 亚洲色图50p| 亚洲2020天天堂在线观看| 精品亚洲va在线va天堂资源站| 欧美日韩亚洲激情| 久久国产精品99国产精| 在线视频国产日韩| 亚洲性av在线| 亚洲欧美国产日韩天堂区| 亚洲免费影视第一页| 色系列之999| www.亚洲一二| 久久免费视频观看| 日韩美女视频在线观看| 久久亚洲私人国产精品va| 91精品国产高清久久久久久91| 欧美日韩另类在线| 国产成人亚洲精品| 国产成人综合精品| 国产精品18久久久久久首页狼| 性欧美暴力猛交69hd| 最新国产成人av网站网址麻豆| www.久久久久久.com| 中文字幕日韩精品有码视频| 日韩的一区二区| 国产精品久久久久久久久久久久| 日韩的一区二区| 日韩欧美中文在线| 亚洲国产婷婷香蕉久久久久久| 欧美在线观看一区二区三区| 久久天天躁狠狠躁夜夜躁| 一二美女精品欧洲| 草民午夜欧美限制a级福利片| 日韩高清电影免费观看完整版| 国产精品亚洲网站| 成人两性免费视频| 91亚洲国产精品| 在线观看视频亚洲| 97在线看福利| 亚洲国产成人精品久久久国产成人一区| 亚洲精品aⅴ中文字幕乱码| 91色在线视频| 久久精品国产欧美亚洲人人爽| 日韩精品视频在线免费观看| 日韩电影中文字幕| 韩国精品久久久999| 国产国产精品人在线视| 在线精品国产欧美| 中文字幕精品网| 欧美中文字幕在线| 精品欧美激情精品一区| 亚洲精品视频播放| 日韩成人在线视频网站| 91亚洲永久免费精品| 亚洲精品第一国产综合精品| 97精品国产97久久久久久春色| 日本高清不卡在线| 久久天天躁狠狠躁夜夜av| 国内精品视频在线| 日本午夜精品理论片a级appf发布| 国产精品精品视频一区二区三区| 日韩av在线免费播放| 美乳少妇欧美精品| 亚洲2020天天堂在线观看| 91在线直播亚洲| 欧美日韩国产精品专区| 欧美亚洲另类视频| 亚洲天堂视频在线观看| 中国日韩欧美久久久久久久久| 国内精品久久久久久久久| 国产免费一区二区三区在线观看| 丝袜亚洲另类欧美重口| 国产精品偷伦视频免费观看国产| 国产精品国产亚洲伊人久久| 国产一区二区香蕉| 国产精品自在线| 亚洲美女免费精品视频在线观看| 精品女同一区二区三区在线播放| 国产亚洲一级高清| 国产精品视频最多的网站| 国产精品高潮呻吟久久av野狼| 青青a在线精品免费观看| 日韩在线视频中文字幕| 久久影视三级福利片| 午夜精品久久17c| 91系列在线播放| 亚洲精品福利在线观看| 国产精品永久免费在线| 国产欧亚日韩视频| 成人国内精品久久久久一区| 国产综合久久久久久| 国产精品福利网站| 亚洲天堂av电影| 伊人伊成久久人综合网小说| 日韩中文字幕在线| 日韩精品在线观看一区| 亚洲精品电影网在线观看| 日本精品性网站在线观看| 亚洲一品av免费观看| 日韩在线免费观看视频| 欧美精品在线极品| 亚洲一区二区三区香蕉| 欧美在线视频a| 在线观看国产欧美| 在线视频国产日韩| 日韩免费在线免费观看| 国产在线98福利播放视频| 国产视频精品免费播放| 午夜精品一区二区三区视频免费看| 亚洲剧情一区二区| 欧美在线一级va免费观看| 欧美黑人视频一区| 91在线看www| 97国产成人精品视频| 92国产精品视频| 久久久久国产精品一区| 91精品国产高清久久久久久91| 国产精品男人爽免费视频1| 中文字幕一精品亚洲无线一区| 亚洲国产第一页| 国产精品无av码在线观看| 欧美激情xxxx| 韩国三级电影久久久久久| 国产伦精品一区二区三区精品视频| 国产欧美亚洲视频| 国产精品爽黄69天堂a|