1. 程式人生 > >向函式中傳遞指標和傳遞指標引用的區別

向函式中傳遞指標和傳遞指標引用的區別

如果是傳遞指標,那麼會先複製該指標,在函式內部使用的是複製後的指標,這個指標與原來的指標指向相同的地址,如果在函式內部將複製後的指標指向了另外的新的物件,那麼不會影響原有的指標;
但 是對於傳遞指標應用,如果將傳遞進來的指標指向了新的物件,那麼原始的指標也就指向了新的物件,這樣就會造成記憶體洩漏,因為原來指標指向的地方已經不能再 引用了,即使沒有將傳遞進來的指標指向新的物件,而是在函式結束的時候釋放了指標,那麼在函式外部就不能再使用原有的指標了,因為原來的記憶體已經被釋放了

看個例子:

#include <cstdlib>
#include <cstring>
#include <memory>
#include <iostream>

using namespace std;

void func2(char *str)
{
    str = (char *)malloc(10);
    strcpy(str, "hello");
}

void func3(char *&str)
{
    str = (char *)malloc(10);
    strcpy(str, "hello");
}
int main()
{
    char *str = "1234567890";

    func2(str);
    cout << str << endl;
    func3(str);
    cout << str << endl;
}

程式輸出:

1234567890
hello

簡單一點可以這麼想,如果不用引用的話,被傳遞的引數本身是不能被修改的, 
即使你傳遞的是指標,也不過能修改指標指向的內容,不能修改指標本身。 


如果要修改當前被傳遞的引數的話,要麼再加一級指標,要麼用引用。

指標的引用類似於二級指標

傳遞指標的話,在函式體內你只能修改指標的內容,不能改變指標本身(非const type *型別,指標本身也能修改的,只是這個修改,不會影響到實參指標) 
傳遞指標的引用,在函式體內,你既可以修改指標的內容,也可以修改指標本身(實參)

有區別, 
無論你傳值還是傳指標,函式都會生成一個臨時變數, 但傳引用時,不會生成臨時變數, 當你傳值時,只可以引用值而不可以改變值,但傳值引用時,可以改變值, 當你傳指標時,只可以改變指標所指的內容,不可以改變指標本身,但傳指標引用時,即可以改變指標所指的內容,又可以改變指標本身,但傳引用主要是它不生成臨時變數,不進行返回值copy等,速度快。

從概念上講。指標從本質上講就是存放變數地址的一個變數,在邏輯上是獨立的,它可以被改變,包括其所指向的地址的改變和其指向的地址中所存放的資料的改變。

而引用是一個別名,它在邏輯上不是獨立的,它的存在具有依附性,所以引用必須在一開始就被初始化,而且其引用的物件在其整個生命週期中是不能被改變的(自始至終只能依附於同一個變數)。

C++中,指標和引用經常用於函式的引數傳遞,然而,指標傳遞引數和引用傳遞引數是有本質上的不同的:

指標傳遞引數本質上是值傳遞的方式,它所傳遞的是一個地址值。值傳遞過程中,被調函式的形式引數作為被調函式的區域性變數處理,即在棧中開闢了記憶體空間以存放由主調函式放進來的實參的值,從而成為了實參的一個副本。值傳遞的特點是被調函式對形式引數的任何操作都是作為區域性變數進行,不會影響主調函式的實參變數的值。

而在引用傳遞過程中,被調函式的形式引數雖然也作為區域性變數在棧中開闢了記憶體空間,但是這時存放的是由主調函式放進來的實參變數的地址。被調函式對形參的任何操作都被處理成間接定址,即通過棧中存放的地址訪問主調函式中的實參變數。正因為如此,被調函式對形參做的任何操作都影響了主調函式中的實參變數。

引用傳遞和指標傳遞是不同的,雖然它們都是在被調函式棧空間上的一個區域性變數,但是任何對於引用引數的處理都會通過一個間接定址的方式操作到主調函式中的相關變數。而對於指標傳遞的引數,如果改變被調函式中的指標地址,它將影響不到主調函式的相關變數。如果想通過指標引數傳遞來改變主調函式中的相關變數,那就得使用指向指標的指標,或者指標引用。

為了進一步加深大家對指標和引用的區別,下面我從編譯的角度來闡述它們之間的區別:

程式在編譯時分別將指標和引用新增到符號表上,符號表上記錄的是變數名及變數所對應地址。指標變數在符號表上對應的地址值為指標變數的地址值,而引用在符號表上對應的地址值為引用物件的地址值。符號表生成後就不會再改,因此指標可以改變其指向的物件(指標變數中的值可以改),而引用物件則不能修改。

最後,總結一下指標和引用的相同點和不同點:

★相同點:

●都是地址的概念;

指標指向一塊記憶體,它的內容是所指記憶體的地址;而引用則是某塊記憶體的別名。

★不同點:

●指標是一個實體,而引用僅是個別名;

●引用只能在定義時被初始化一次,之後不可變;指標可變;引用“從一而終”,指標可以“見異思遷”;

●引用沒有const,指標有constconst的指標不可變;

●引用不能為空,指標可以為空;

●“sizeof 引用”得到的是所指向的變數(物件)的大小,而“sizeof 指標”得到的是指標本身的大小;

●指標和引用的自增(++)運算意義不一樣;

●引用是型別安全的,而指標不是 (引用比指標多了型別檢查