c 值傳遞 位址傳遞 引用傳遞

2021-06-10 08:05:57 字數 3034 閱讀 6700

一.函式引數傳遞方式之一:值傳遞

1. 值傳遞的乙個錯誤認識

先看題一中exchg1函式的定義:

void exchg1(int x, int y)   //定義中的x,y變數被稱為exchg1函式的形式引數

問:你認為這個函式是在做什麼呀?

答:好像是對引數x,y的值對調吧?

請往下看,我想利用這個函式來完成對a,b兩個變數值的對調,程式如下:

void main()

我問: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呢?

奇怪,明明我把a,b分別代入了x,y中,並在函式裡完成了兩個變數值的交換,為什麼a,b變數值還是沒有交換(仍然是a==4,b==6,而不是a==6,b==4)?如果你也會有這個疑問,那是因為你跟本就不知實參a,b與形參x,y的關係了。

2. 乙個預備的常識

為了說明這個問題,我先給出乙個**:

int a=4;

int x;

x=a;

x=x+3;

看好了沒,現在我問你:最終a值是多少,x值是多少?

(怎麼搞的,給我這個小兒科的問題。還不簡單,不就是a==4  x==7嘛!)

在這個**中,你要明白乙個東西:雖然a值賦給了x,但是a變數並不是x變數哦。我們對x任何的修改,都不會改變a變數。呵呵!雖然簡單,並且一看就理所當然,不過可是乙個很重要的認識喔。

3. 理解值傳遞的形式

看呼叫exch1函式的**:

main()

exchg1(a,b)時所完成的操作**如下所示。

int x=a;//←

int y=b;//←注意這裡,頭兩行是呼叫函式時的隱含操作

int tmp;

tmp=x;

x=y;

y=tmp;

請注意在呼叫執行exchg1函式的操作中我人為地加上了頭兩句:

int x=a;

int y=b;

這是呼叫函式時的兩個隱含動作。它確實存在,現在我只不過把它顯式地寫了出來而已。問題一下就清晰起來啦。(看到這裡,現在你認為函式裡面交換操作的是a,b變數或者只是x,y變數呢?)

原來 ,其實函式在呼叫時是隱含地把實參a,b 的值分別賦值給了x,y,之後在你寫的exchg1函式體內再也沒有對a,b進行任何的操作了。交換的只是x,y變數。並不是a,b。當然a,b的值沒有改變啦!函式只是把a,b的值通過賦值傳遞給了x,y,函式裡頭操作的只是x,y的值並不是a,b的值。這就是所謂的引數的值傳遞了。

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

二. 函式引數傳遞方式之二:位址傳遞

繼續——位址傳遞的問題!

看題二的**:

exchg2(int *px, int *py)

main()

它的輸出結果是:

*px=6,*py=4

a=6,b=4

看函式的介面部分:exchg2(int *px,int *py),請注意:引數px,py都是指標。

再看呼叫處:exchg2(&a, &b);

它將a的位址(&a)代入到px,b的位址(&b)代入到py。同上面的值傳遞一樣,函式呼叫時作了兩個隱含的操作:將&a,&b的值賦值給了px,py。

px=&a;

py=&b;

呵呵!我們發現,其實它與值傳遞並沒有什麼不同,只不過這裡是將a,b的位址值傳遞給了px,py,而不是傳遞的a,b的內容,而(請好好地在比較比較啦)

整個exchg2函式呼叫是如下執行的:

px=&a;   //

py=&b;   //請注意這兩行,它是呼叫exchg2的隱含動作。

int tmp=*px;

*px=*py;

*py=tmp;

print(「*px=%d,*py=%d\n」,*px,*py);

這樣,有了頭兩行的隱含賦值操作。我們現在已經可以看出,指標px,py的值已經分別是a,b變數的位址值了。接下來,對*px,*py的操作當然也就是對a,b變數本身的操作了。所以函式裡頭的交換就是對a,b值的交換了,這就是所謂的位址傳遞(傳遞a,b的位址給了px,py),你現在明白了嗎?

三. 函式引數傳遞方式之三:引用傳遞

看題三的**:

exchg3(int &x, int &y) //注意定義處的形式引數的格式與值傳遞不同

main()

輸出結果:

x=6, y=4

a=6, b=4   //這個輸出結果與值傳遞不同。

看到沒有,與值傳遞相比,**格式上只有一處是不同的,即在定義處:

exchg3(int &x, int &y)。

但是我們發現a與b的值發生了對調。這說明了exchg3(a,b)裡頭修改的是a,b變數,而不只是修改x,y了。

我們先看exchg3函式的定義處exchg3(int &x,int &y)。

相當於int &x=a;

int &y=b;

temp=x;

x=y;

y=temp;

x是a的引用(別名),對x的操作相當於對a的操作。

最後對值傳遞與引用傳遞作乙個比較:

1. 在函式定義格式上有不同:

值傳遞在定義處是:exchg1(int x, int y);

引用傳遞在這義處是:exchg1(int &x, int &y);

2. 呼叫時有相同的格式:

值傳遞:exchg1(a,b);

引用傳遞:exchg3(a,b);

3. 功能上是不同的:

值傳遞的函式裡操作的不是a,b變數本身,只是將a,b值賦給了x,y函式裡操作的只是x,y變數而不是a,b,顯示a,b的值不會被exchg1函式所修改。

引用傳遞exchg3(a,b)函式裡是用a,b分別代替了x,y。函式裡操作的是a,b。

C 值傳遞 位址傳遞 引用傳遞

昨天遇到乙個問題 程式一 char str 256 char sp str visitdepart rs,rs.column,sp static void visitdepart result t rs,int num,char spx note 這個程式表面上看是沒有錯誤的,但是,我用它做了樹的深...

值傳遞 位址傳遞 引用傳遞

值傳遞使得實參與形參沒有聯絡,函式與函式之間沒有耦合性,在各自的函式裡修改值不會改變各自本身的值。但這也是它的缺陷。1.實參傳入被調函式無法交換數值。2.當傳入值大於8位元組即需要進行整體拷貝,無論是開闢的空間還是資料的拷貝都相對較大 這也是設計陣列退化為指標的原因 而指標傳遞只有4個位元組 所以需...

值傳遞 引用傳遞與位址傳遞對比

引用傳遞 形參與實參公用乙個記憶體單元,同時變化。值傳遞 形參是實參的轉殖體,形參在記憶體單元中發生變化不會影響到實參。形參變數只有在被呼叫時才分配記憶體單元,在呼叫結束時,即刻釋放所分配的。記憶體單元。因此,形參只在函式內部有效。函式呼叫結束返回主呼叫函式後則不能再使用該形參變數。實參可以是常量 ...