C++編程語言從出現至今已經發展的越發成熟,逐漸成為了開發領域中一個重要的應用語言。今天大家可以從這篇文章中詳細了解到有關C++頭文件的一些嵌套方法,從而讓大家更進一步的對這一語言有一個詳細的了解。
在實際編程中,不同的類一般是放在不同的相互獨立的C++頭文件中的,這樣兩個類在相互引用時又會有不一樣的問題。重復編譯是問題出現的根本原因。為了保證頭文 件僅被編譯一次,在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()
- {
- ......
- }
編譯文件,會出現class type redefinition的錯誤
為什麼會出現類重復定義的錯誤呢?請讀者仔細查看EX10.cpp文件,在這個文件中包含了animal.h和fish.h這兩個頭文件。當編譯器編譯EX10.cpp文件時,因為在文件中包含了animal.h頭文件,編譯器展開這個C++頭文件,知道animal這個類定義了,接著展開fish.h 頭文件,而在fish.h頭文件中也包含了animal.h,再次展開animal.h,於是animal這個類就重復定義了。
要解決C++頭文件重復包含的問題,可以使用條件預處理指令。修改後的頭文件如下:
- 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
我們再看EX10.cpp的編譯過程。當編譯器展開animal.h頭文件時,條件預處理指令判斷ANIMAL_H_H沒有定義,於是就定 義它,然後繼續執行,定義了animal這個類;接著展開fish.h頭文件,而在fish.h頭文件中也包含了animal.h,再次展開 animal.h,這個時候條件預處理指令發現ANIMAL_H_H已經定義,於是跳轉到#endif,執行結束。
但是不要以為使用了這種機制就全部搞定了,比如在以下的代碼中:
- //文件A.h中的代碼
- #pragma once
- #include "B.h"
- class A
- {
- public:
- B* b;
- };
- //文件B.h中的代碼
- #pragma once
- #include "A.h"
- class B
- {
- public:
- A* a;
- };
這裡兩者都使用了指針成員,因此嵌套本身不會有什麼問題,在主函數前面使用#include "A.h"之後,主要編譯錯誤如下:
- error C2501: 'A' : missing storage-class or type specifiers
仍然是類型不能找到的錯誤。其實這裡仍然需要前置聲明。分別添加前置聲明之後,可以成功編譯了。代碼形式如下:
- //文件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;
- };
這樣至少可以說明,C++頭文件包含代替不了前置聲明。有的時候只能依靠前置聲明來解決問題。我們還要思考一下,有了前置聲明的時候頭文件包含還是必要的 嗎?我們嘗試去掉A.h和B.h中的#include行,發現沒有出現新的錯誤。那麼究竟什麼時候需要前置聲明,什麼時候需要頭文件包含呢?