1 引言
舉個例子: 在func函數退出后,指針pInt所指的內容*pInt為 12
#include <stdio.h>
//公眾號:C語言與CPP編程
int func(int* pRes)
{
if(pRes == NULL)
pRes = new int(12);//分配新的內存空間給指針pRes,并賦值
return 0;
}
int main ()
{
int *pInt = NULL;
int val = func(pInt);
printf("%dn",*pInt); return 0;
}
解析:int func(int* pRes)函數的形參是指針類型 int *pRes,在函數體中 new了一塊內存并賦值 12,將內存地址賦值給指針 pRes。在main函數中,定義了指針pInt,調用func函數,把pInt作為參數傳入func函數中。結果*pInt并不是 12。
原因:在func函數調用過程中,形參和實參的傳遞使用了值傳遞方式,這種情況下,形參變量在函數體內發生了變化,在函數結束之后,形參變量隨之釋放,不能把變化的結果返回給實參。
可以使用指針傳遞或者引用傳遞。想要在函數體內改變pRes的值,并把這個變化返回到main函數中,必須傳遞pRes的指針。因為pRes本身就是指針,所以應該傳遞指針的指針,或者指針的引用。
指針的引用
int v = 1;
int *p = &v;'
int *&rp = p;
&說明r是一個引用。*確定r引用的類型是一個指針。
因為引用不是對象,故無引用的數組,無指向引用的指針,無到引用的引用:
int& a[5]; // 錯誤
int&* p; // 錯誤
int& &r; // 錯誤
所以修改函數int func(int* pRes);為int func(int *&pRes);
#include <stdio.h>
int func(int* &pRes)
{
if(pRes == NULL)
pRes = new int(12);//分配新的內存空間給指針pRes,并賦值
return 0;
}
int main ()
{
int *pInt = NULL;
int val = func(pInt);
printf("%dn",*pInt);
return 0;
}
2 傳值、傳引用區別和聯系
傳值:實參拷貝傳遞給形參。就是把實參賦值給形參,賦值完畢后實參就和形參沒有任何聯系,對形參的修改就不會影響到實參。
傳地址:把實參地址的拷貝傳遞給形參。就是把實參的地址復制給形參。復制完畢后實參的地址和形參的地址沒有任何聯系,對實參形參地址的修改不會影響到實參, 但是對形參地址所指向對象的修改卻直接反應在實參中,因為形參指向的對象就是形參的對象。
傳引用:本質沒有任何實參的拷貝,兩個變量指向同一個對象。這是對形參的修改,必然反映到實參上。
- 無論傳值還是傳指針,函數都會生成一個臨時變量,但傳引用時,不會生成臨時變量,
- 傳值時,只可以引用值而不可以改變值,但傳值引用時,可以改變值,
- 傳指針時,只可以改變指針所指的內容,不可以改變指針本身,但傳指針引用時,既可以改變指針所指的內容,又可以改變指針本身,
- 引用傳遞函數的參數,在內存中并沒有產生實參的副本,它是直接對實參操作;而使用一般變量傳遞函數的參數,當發生函數調用時,需要給形參分配存儲單元,形參變量是實參變量的副本;如果傳遞的是對象,還將調用拷貝構造函數。因此,當參數傳遞的數據較大時,用引用比用一般變量傳遞參數的效率和所占空間都好。