作者khoguan (Khoguan Phuann)
看板C_and_CPP
標題[FAQ集] 無標頭檔,直接呼叫 printf() 可以嗎?
時間Thu Sep 1 23:39:15 2005
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
1F:推 prudent:說的好~~ 頂一個... 220.142.39.22 09/02
※ 編輯: khoguan 來自: 220.130.208.167 (09/03 13:31)