如果把函數比喻成一台機器,那么參數就是原材料,返回值就是最終產品;從一定程度上講,函數的作用就是根據不同的參數產生不同的返回值。
這一節我們先來講解C語言函數的參數,下一節再講解C語言函數的返回值。
C語言函數的參數會出現在兩個地方,分別是函數定義處和函數調用處,這兩個地方的參數是有區別的。
形參(形式參數)
在函數定義中出現的參數可以看做是一個占位符,它沒有數據,只能等到函數被調用時接收傳遞進來的數據,所以稱為形式參數,簡稱形參。
這里按自己的理解為什么說形參看作一個占位符,后文還說到在調用之前並沒有給他分配內存,看上去這兩段是有些矛盾的,其實機器在時間上的運行順序這么說沒毛病;
本人理解是在程序編譯過程中,程序計數器不斷加以程序順序執行,執行到調用函數位置時,會將之前的數據壓棧就像中斷一樣,調用前要傳遞實參值對應的內存位置並沒有開辟,所以可以說形參再調用前不占內存;
但是這里沒有開辟不是說實參會傳到隨機的位置,這是不會發生的,程序在編譯好之后,每執行一條指令,數據都會有個確切的內存位置對應(這是在編譯器編譯的時候就已經確定好的),所以這么說來,形參確實是一個占位符(占有一個內存地址的位置);
只是調用前別的參數也可以占用這個位置,但編譯器編譯的時候肯定不會讓一個全局變量的內存地址和分配給形參的內存地址是同一個地址,原因大家應該通過上面的解釋也明白編譯器在編譯時不會讓兩個沒有關聯的變量,同時服務於一個地址這樣會造成數據的混亂。
就像兩個服務員負責同一桌客人時,當客人需要一瓶飲料時,兩個服務員分別在不同時刻聽到,可能就會造成給客人拿兩瓶情況。
實參(實際參數)
函數被調用時給出的參數包含了實實在在的數據,會被函數內部的代碼使用,所以稱為實際參數,簡稱實參。
形參和實參的功能是傳遞數據,發生函數調用時,實參的值會傳遞給形參。
形參和實參的區別和聯系
1) 形參變量只有在函數被調用時才會分配內存,調用結束后,立刻釋放內存,所以形參變量只有在函數內部有效,不能在函數外部使用。
2) 實參可以是常量、變量、表達式、函數等,無論實參是何種類型的數據,在進行函數調用時,它們都必須有確定的值,以便把這些值傳送給形參,所以應該提前用賦值、輸入等辦法使實參獲得確定值。
3) 實參和形參在數量上、類型上、順序上必須嚴格一致,否則會發生“類型不匹配”的錯誤。當然,如果能夠進行自動類型轉換,或者進行了強制類型轉換,那么實參類型也可以不同於形參類型。
4) 函數調用中發生的數據傳遞是單向的,只能把實參的值傳遞給形參,而不能把形參的值反向地傳遞給實參;換句話說,一旦完成數據的傳遞,實參和形參就再也沒有瓜葛了,所以,在函數調用過程中,形參的值發生改變並不會影響實參。
請看下面的例子:
1 #include <stdio.h> 2 //計算從m加到n的值 3 int sum(int m, int n) 4 { 5 int i; 6 for (i = m+1; i <= n; ++i) 7 { 8 m += i; 9 } 10 return m; 11 } 12 int main() 13 { 14 int a, b, total; 15 printf("Input two numbers: "); 16 scanf("%d %d", &a, &b); 17 total = sum(a, b); 18 printf("a=%d, b=%d\n", a, b); 19 printf("total=%d\n", total); 20 return 0; 21 }
運行結果:
Input two numbers: 1 100↙
a=1, b=100
total=5050
在這段代碼中,函數定義處的 m、n 是形參,函數調用處的 a、b 是實參。通過 scanf() 可以讀取用戶輸入的數據,並賦值給 a、b,在調用 sum() 函數時,這份數據會傳遞給形參 m、n。
從運行情況看,輸入 a 值為 1,即實參 a 的值為 1,把這個值傳遞給函數 sum() 后,形參 m 的初始值也為 1,在函數執行過程中,形參 m 的值變為 5050。函數運行結束后,輸出實參 a 的值仍為 1,可見實參的值不會隨形參的變化而變化。
以上調用 sum() 時是將變量作為函數實參,除此以外,你也可以將常量、表達式、函數返回值作為實參,如下所示:
- total = sum(10, 98); //將常量作為實參
- total = sum(a+10, b-3); //將表達式作為實參
- total = sum( pow(2,2), abs(-100) ); //將函數返回值作為實參
5) 形參和實參雖然可以同名,但它們之間是相互獨立的,互不影響,因為實參在函數外部有效,而形參在函數內部有效。
更改上面的代碼,讓實參和形參同名:
1 #include <stdio.h> 2 //計算從m加到n的值 3 int sum(int m, int n) 4 { 5 int i; 6 for (i = m + 1; i <= n; ++i) 7 { 8 m += i; 9 } 10 return m; 11 } 12 int main() 13 { 14 int m, n, total; 15 printf("Input two numbers: "); 16 scanf("%d %d", &m, &n); 17 total = sum(m, n); 18 printf("m=%d, n=%d\n", m, n); 19 printf("total=%d\n", total); 20 return 0; 21 }
運行結果:
Input two numbers: 1 100
m=1, n=100
total=5050
調用 sum() 函數后,函數內部的形參 m 的值已經發生了變化,而函數外部的實參 m 的值依然保持不變,可見它們是相互獨立的兩個變量,除了傳遞參數的一瞬間,其它時候是沒有瓜葛的。