[FAQ集] 無標頭檔,直接呼叫 printf() 可以嗎?

看板C_and_CPP (C/C++)作者 (Khoguan Phuann)時間19年前 (2005/09/01 23:33), 編輯推噓1(100)
留言1則, 1人參與, 最新討論串1/1
ptt.cc BBS 站 C_and_CPP 板 FAQ (0.1版) 函式語法 Q: 有一次我忘了 #include <stdio.h> 就呼叫 printf() 竟然也可以,何故? A: 這是 C 比較不嚴謹的地方。 當我們呼叫一個函式 function_name()時,若 在呼叫處所在的範疇(scope)中,不存在相應的宣告,那麼 C 便會假定,包 含這個函式呼叫的最內層範疇中,存在這樣的宣告: extern int function_name(); 對於這個隱式(implicit)的宣告,三點值得注意: 一、它具有外部連結。而且它的實際定義,存在於別處(也許編程者自己定 義在同一個原始碼檔的下方,也許存在於標準函式庫中)。 二、它的傳回值型別一律是 int。 三、它的參數未指定。(其後果參見上則) 所以如果我們在呼叫標準函式時,未能適當的引入標頭檔就直接呼叫。那麼 編譯器就不再為我們根據標頭檔的函式宣告來做檢查,也不會默默的替我們 去找標頭檔,只會根據上述的隱式宣告來編譯這個源碼檔。最後在連結時, 連結器卻會去標準函式庫中硬生生的連結同名的函式。這樣一來,即使我們 呼叫時傳入的參數個數、型別,或是使用傳回值的方式不合標準函式庫那個 同名程式,一樣會連結成功,產生一個不知會跑出什麼結果的可執行檔。 例如: int main() { int s = 0; printf("s==%d\n", s); /* extern int printf() */ s = srand(1999); /* extern int srand() 和實際定義有出入 */ printf("s=="%d\n", s); return 0; } 其實 srand() 的傳回型別是 void, 在這裡卻賦值給一個 int。因為沒引入 標頭檔,編譯器就失去為我們把關的能力。連結成功,執行的結果,在不同 系統上,會得到不同結果。 至於 C++ 在這方面就嚴謹多了,沒有先明文宣告(包括引入合適的標頭檔), 是不能直接呼叫函式的。 呼叫一個並未明文宣告的函式是不良的 C 程式風格,更是錯誤的 C++ 程式。 結論就是,無論寫 C 或 C++,都應該規規矩矩的,該 include 的就 include。 一時偷懶,要除錯時就有得受了。 ----------------------------------- 敬請指正錯誤,或提出更合適的答案。 -- ※ 發信站: 批踢踢實業坊(ptt.cc) ◆ From: 220.130.208.167

220.142.39.22 09/02, , 1F
說的好~~ 頂一個...
220.142.39.22 09/02, 1F
※ 編輯: khoguan 來自: 220.130.208.167 (09/03 13:31)
文章代碼(AID): #135nzEF4 (C_and_CPP)
文章代碼(AID): #135nzEF4 (C_and_CPP)