C++頭文件如何進(jìn)行重用
C++編程語(yǔ)言從出現(xiàn)至今已經(jīng)發(fā)展的越發(fā)成熟,逐漸成為了開(kāi)發(fā)領(lǐng)域中一個(gè)重要的應(yīng)用語(yǔ)言。今天大家可以從這篇文章中詳細(xì)了解到有關(guān)C++頭文件的一些嵌套方法,從而讓大家更進(jìn)一步的對(duì)這一語(yǔ)言有一個(gè)詳細(xì)的了解。
在實(shí)際編程中,不同的類一般是放在不同的相互獨(dú)立的C++頭文件中的,這樣兩個(gè)類在相互引用時(shí)又會(huì)有不一樣的問(wèn)題。重復(fù)編譯是問(wèn)題出現(xiàn)的根本原因。為了保證頭文 件僅被編譯一次,在C++中常用的辦法是使用條件編譯命令。
- Example:
- animal.h
- class animal
- {
- ......
- };
- animal.cpp
- #include "animal.h"
- #include <iostream.h>
- ......
- fish.h
- #include "animal.h"
- class fish
- {
- ......
- };
- fish.cpp
- #include "fish.h"
- #include <iostream.h>
- ......
- main.cpp
- #include "animal.h"
- #include "fish.h"
- void main()
- {
- ......
- }
編譯文件,會(huì)出現(xiàn)class type redefinition的錯(cuò)誤
為什么會(huì)出現(xiàn)類重復(fù)定義的錯(cuò)誤呢?請(qǐng)讀者仔細(xì)查看EX10.cpp文件,在這個(gè)文件中包含了animal.h和fish.h這兩個(gè)頭文件。當(dāng)編譯器編譯EX10.cpp文件時(shí),因?yàn)樵谖募邪薬nimal.h頭文件,編譯器展開(kāi)這個(gè)C++頭文件,知道animal這個(gè)類定義了,接著展開(kāi)fish.h 頭文件,而在fish.h頭文件中也包含了animal.h,再次展開(kāi)animal.h,于是animal這個(gè)類就重復(fù)定義了。
要解決C++頭文件重復(fù)包含的問(wèn)題,可以使用條件預(yù)處理指令。修改后的頭文件如下:
- animal.h
- #ifndef ANIMAL_H_H
- #define ANIMAL_H_H
- class animal
- {
- ......
- };
- #endif
- fish.h
- #include "animal.h"
- #ifndef FISH_H_H
- #define FISH_H_H
- class fish
- {
- ......
- };
- #endif
我們?cè)倏碋X10.cpp的編譯過(guò)程。當(dāng)編譯器展開(kāi)animal.h頭文件時(shí),條件預(yù)處理指令判斷ANIMAL_H_H沒(méi)有定義,于是就定 義它,然后繼續(xù)執(zhí)行,定義了animal這個(gè)類;接著展開(kāi)fish.h頭文件,而在fish.h頭文件中也包含了animal.h,再次展開(kāi) animal.h,這個(gè)時(shí)候條件預(yù)處理指令發(fā)現(xiàn)ANIMAL_H_H已經(jīng)定義,于是跳轉(zhuǎn)到#endif,執(zhí)行結(jié)束。
但是不要以為使用了這種機(jī)制就全部搞定了,比如在以下的代碼中:
- //文件A.h中的代碼
- #pragma once
- #include "B.h"
- class A
- {
- public:
- B* b;
- };
- //文件B.h中的代碼
- #pragma once
- #include "A.h"
- class B
- {
- public:
- A* a;
- };
這里兩者都使用了指針成員,因此嵌套本身不會(huì)有什么問(wèn)題,在主函數(shù)前面使用#include "A.h"之后,主要編譯錯(cuò)誤如下:
- error C2501: 'A' : missing storage-class or type specifiers
仍然是類型不能找到的錯(cuò)誤。其實(shí)這里仍然需要前置聲明。分別添加前置聲明之后,可以成功編譯了。代碼形式如下:
- //文件A.h中的代碼
- #pragma once
- #include "B.h"
- class B;
- class A
- {
- public:
- B* b;
- };
- //文件B.h中的代碼
- #pragma once
- #include "A.h"
- class A;
- class B
- {
- public:
- A* a;
- };
這樣至少可以說(shuō)明,C++頭文件包含代替不了前置聲明。有的時(shí)候只能依靠前置聲明來(lái)解決問(wèn)題。我們還要思考一下,有了前置聲明的時(shí)候頭文件包含還是必要的 嗎?我們嘗試去掉A.h和B.h中的#include行,發(fā)現(xiàn)沒(méi)有出現(xiàn)新的錯(cuò)誤。那么究竟什么時(shí)候需要前置聲明,什么時(shí)候需要頭文件包含呢?
【編輯推薦】