想必很多人都看過“頭文件中用到的 #ifndef/#define/#endif 來防止該頭文件被重復引用”。但是是否能理解“被重復引用”是什麼意思?頭文件被重復引用了,會產生什麼後果?是不是所有的頭文件中都要加入#ifndef/#define/#endif 這些代碼?
1、 其實“被重復引用”是指一個頭文件在同一個cpp文件中被include了多次,這種錯誤常常是由於include嵌套造成的。如:存在a.h文件#include "c.h"而此時b.cpp文件導入了#include "a.h" 和#include "c.h"此時就會造成c.h重復包含。
2、頭文件被重復引用引起的後果:
(1)有些頭文件重復引用,只是增加了編譯工作的工作量,不會引起太大的問題,僅僅是編譯效率低一些,但是對於大工程而言編譯效率就是很重要的了。
(2)有些頭文件重復包含,會引起編譯錯誤,比如在頭文件中定義了全局變量或寫了函數的實現而不是聲明(雖然這種方式不被推薦,但確實是C規范允許的),這種會引起重復定義。
3、 是不是所有的頭文件中都要加入這些代碼?
不是一定要加,但是不管怎樣,用#ifndef/#define/#endif或者其他方式避免頭文件重復包含,只有好處沒有壞處。培養一個好的編程習慣是學習編程的一個重要分支。所以在寫頭文件時,最好是把內容都寫在#ifndef和#endif之間。
下面給出#ifndef/#define/#endif的用法:
#ifndef __XXX_H__ //意思是 "if not define __XXX_H__" 也就是沒包含XXX.h #define __XXX_H__ //就定義__XXX_H__ ... //此處放頭文件中本來應該寫的代碼 #endif //否則不需要定義
若未定義XXX.h則這裡就定義XXX.h,然後運行裡面的內容,若下次還走到了這個文件,則進行#ifndef的判斷,則#ifndef與#endif之間的內容不會再次被載入
但是,必須記住的是預處理器仍將整個頭文件讀入,即使這個頭文件所有內容將被忽略。由於這種處理將減慢編譯速度,所以如果可能,應該避免出現多重包含。
補充:
1、#pragma的用法
#pragma once ... //此處放頭文件中本來應該寫的代碼
#pragma once 是個預處理指令,在頭文件的最開始加入這條指令表示:這個頭文件只被編譯一次,是由編譯器提供保證:同一個文件不會被包含多次。注意這裡所說的“同一個文件”是指物理上的一個文件,而不是指內容相同的兩個文件。帶來的好處是,你不必再費勁想個宏名了,當然也就不會出現宏名碰撞引發的奇怪問題。
總結:
#ifndef,#define,#endif是C/C++語言中的宏定義,通過宏定義避免文件多次編譯。所以在所有支持C++語言的編譯器上都是有效的,移植性好,所以如果寫的程序要跨平台,最好使用這種方式。但缺點是宏名字不能沖突。
#pragma 可以避免名字沖突,缺點就是如果某個頭文件有多份拷貝,本方法不能保證他們不被重復包含。且不是所有編譯器都支持這種方式。