函數參數傳遞方式

函數參數傳遞方式之一:值傳遞

1)值傳遞的一個錯誤認識
先看考題一中Exchg1函數的定義:
void Exchg1(int x, int y) /*
定義中的x,y變量被稱爲Exchg1函數的形式參數 */
{
   inttmp;
   tmp = x;
   x = y;
   y = tmp;
   printf("x = %d, y = %d.\n", x, y);
}
問:你認爲這個函數是在做什麼呀?
答:好像是對參數xy的值對調吧?
請往下看,我想利用這個函數來完成對a,b兩個變量值的對調,程序如下:
main()
{
   int a = 4,b = 6;
   Exchg1(a, b); /*a,b
變量爲Exchg1函數的實際參數。*/
   printf("a = %d, b = %d.\n”, a, b);
   return(0);
}
我問:Exchg1()裏頭的printf("x= %d, y = %d.\n", x, y);語句會輸出什麼啊?我再問:Exchg1()後的printf("a = %d, b = %d.\n”, a, b);語句輸出的是什麼?
程序輸出的結果是:
x = 6, y = 4.
a = 4, b = 6.
爲什麼不是a= 6,b = 4呢?奇怪,明明我把ab分別代入了xy中,並在函數裏完成了兩個變量值的交換,爲什麼ab變量值還是沒有交換(仍然是a = 4b = 6,而不是a = 6b = 4)?如果你也會有這個疑問,那是因爲你根本就不知實參ab與形參xy的關係了。

2)一個預備的常識
爲了說明這個問題,我先給出一個代碼:
   int a = 4;
   int x;
   x = a;
   x = x + 3;
看好了沒,現在我問你:最終a值是多少,x值是多少?
(怎麼搞的,給我這個小兒科的問題。還不簡單,不就是a = 4x = 7嘛!)
在這個代碼中,你要明白一個東西:雖然a值賦給了x,但是a變量並不是x變量哦。我們對x任何的修改,都不會改變a變量。呵呵!雖然簡單,並且一看就理所當然,不過可是一個很重要的認識喔。

3)理解值傳遞的形式
看調用Exch1函數的代碼:
main()
{
   int a = 4,b = 6;
   Exchg1(a, b) /*
這裏調用了Exchg1函數 */
   printf("a = %d, b = %d.\n", a, b);
}
Exchg1(a, b)
時所完成的操作代碼如下所示。
int x = a; /*
*/
int y = b; /*
注意這裏,頭兩行是調用函數時的隱含操作 */
inttmp;
tmp = x;
x = y;
y = tmp;
請注意在調用執行Exchg1函數的操作中我人爲地加上了頭兩句:
   int x = a;
   int y = b;
這是調用函數時的兩個隱含動作。它確實存在,現在我只不過把它顯式地寫了出來而已。問題一下就清晰起來啦。(看到這裏,現在你認爲函數裏面交換操作的是ab變量或者只是xy變量呢?)

原來,其實函數在調用時是隱含地把實參ab 的值分別賦值給了xy,之後在你寫的Exchg1函數體內再也沒有對ab進行任何的操作了。交換的只是xy變量。並不是ab。當然ab的值沒有改變啦!函數只是把ab的值通過賦值傳遞給了xy,函數裏頭操作的只是xy的值並不是ab的值。這就是所謂的參數的值傳遞了。

哈哈,終於明白了,正是因爲它隱含了那兩個的賦值操作,才讓我們產生了前述的迷惑(以爲ab已經代替了xy,對xy的操作就是對ab的操作了,這是一個錯誤的觀點啊!)。

三、函數參數傳遞方式之二:地址傳遞

繼續!地址傳遞的問題!
看考題二的代碼:
void Exchg2(int *px, int *py)
{
   inttmp = *px;
   *px = *py;
   *py = tmp;
   printf("*px = %d, *py = %d.\n", *px, *py);
}
main()
{
   int a = 4;
   int b = 6;
   Exchg2(&a, &b);
   printf("a = %d, b = %d.\n”, a, b);
   return(0);
}
它的輸出結果是:
*px = 6, *py = 4.
a = 6, b = 4.
看函數的接口部分:Exchg2(int*px, int *py),請注意:參數pxpy都是指針。再看調用處:Exchg2(&a, &b);
它將a的地址(&a)代入到pxb的地址(&b)代入到py。同上面的值傳遞一樣,函數調用時作了兩個隱含的操作:將&a&b的值賦值給了pxpy
   px = &a;
   py = &b;
呵呵!我們發現,其實它與值傳遞並沒有什麼不同,只不過這裏是將ab的地址值傳遞給了pxpy,而不是傳遞的ab的內容,而(請好好地在比較比較啦)整個Exchg2函數調用是如下執行的:
   px = &a; /*
*/
   py = &b; /*
請注意這兩行,它是調用Exchg2的隱含動作。*/
   inttmp = *px;
   *px = *py;
   *py = tmp;
   printf("*px =%d, *py = %d.\n", *px, *py);
這樣,有了頭兩行的隱含賦值操作。我們現在已經可以看出,指針pxpy的值已經分別是ab變量的地址值了。接下來,對*px*py的操作當然也就是對ab變量本身的操作了。所以函數裏頭的交換就是對ab值的交換了,這就是所謂的地址傳遞(傳遞ab的地址給了pxpy),你現在明白了嗎?

四、函數參數傳遞方式之三:引用傳遞

看題三的代碼:
void Exchg3(int&x, int&y) /*
注意定義處的形式參數的格式與值傳遞不同 */
{
   inttmp = x;x = y;
   y = tmp;
   printf("x = %d, y = %d.\n", x, y);
}
main()
{
   int a = 4;
   int b = 6;
   Exchg3(a, b); /*
注意:這裏調用方式與值傳遞一樣*/
   printf("a = %d, b = %d.\n”, a, b);
}
輸出結果:
x = 6, y = 4.
a = 6, b = 4. /*
這個輸出結果與值傳遞不同。*/
看到沒有,與值傳遞相比,代碼格式上只有一處是不同的,即在定義處:
   Exchg3(int&x, int&y)
但是我們發現ab的值發生了對調。這說明了Exchg3(a, b)裏頭修改的是ab變量,而不只是修改xy了。

我們先看Exchg3函數的定義處Exchg3(int&x, int&y)。參數xyint的變量,調用時我們可以像值傳遞(如: Exchg1(a, b); )一樣調用函數(如: Exchg3(a, b);)。但是xy前都有一個取地址符號“&”。有了這個,調用Exchg3時函數會將ab 分別代替了xy了,我們稱:xy分別引用了ab變量。這樣函數裏頭操作的其實就是實參ab本身了,也就是說函數裏是可以直接修改到ab的值了。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章