引用和指標的區別

2021-07-06 11:22:52 字數 2636 閱讀 4669

指標與引用看上去完全不同(指標用操作符』*』和』->』,引用使用操作符』.』),但是它們似乎有相同的功能。指標與引用都是讓你間接引用其他物件。你如何決定在什麼時候使用指標,在什麼時候使用引用呢? 

首先,要認識到在任何情況下都不能用指向空值的引用。乙個引用必須總是指向某些物件。因此如果你使用乙個變數並讓它指向乙個物件,但是該變數在某些時候也可能不指向任何物件,這時你應該把變數宣告為指標,因為這樣你可以賦空值給該變數。相反,如果變數肯定指向乙個物件,例如你的設計不允許變數為空,這時你就可以把變數宣告為引用。 

「但是,請等一下」,你懷疑地問,「這樣的**會產生什麼樣的後果?」 

char *pc = 0; // 設定指標為空值 

char& rc = *pc; // 讓引用指向空值 

這是非常有害的,毫無疑問。結果將是不確定的(編譯器能產生一些輸出,導致任何事情都有可能發生),應該躲開寫出這樣**的人除非他們同意改正錯誤。如果你擔心這樣的**會出現在你的軟體裡,那麼你最好完全避免使用引用,要不然就去讓更優秀的程式設計師去做。我們以後將忽略乙個引用指向空值的可能性。 

因為引用肯定會指向乙個物件,在c裡,引用應被初始化。 

string& rs; // 錯誤,引用必須被初始化 

string s("xyzzy"); 

string& rs = s; // 正確,rs指向s 

指標沒有這樣的限制。 

string *ps; // 未初始化的指標 

// 合法但危險 

不存在指向空值的引用這個事實意味著使用引用的**效率比使用指標的要高。因為在使用引用之前不需要測試它的合法性。 

void printdouble(const double& rd) 

// 肯定指向乙個double值 

相反,指標則應該總是被測試,防止其為空: 

void printdouble(const double *pd)  } 

指標與引用的另乙個重要的不同是指標可以被重新賦值以指向另乙個不同的物件。但是引用則總是指向在初始化時被指定的物件,以後不能改變。 

string s1("nancy"); 

string s2("clancy"); 

string& rs = s1; // rs 引用 s1 

string *ps = &s1; // ps 指向 s1 

rs = s2; // rs 仍舊引用s1, 

// 但是 s1的值現在是 

// "clancy" 

ps = &s2; // ps 現在指向 s2; 

// s1 沒有改變 

總的來說,在以下情況下你應該使用指標,一是你考慮到存在不指向任何物件的可能(在這種情況下,你能夠設定指標為空),二是你需要能夠在不同的時刻指向不同的物件(在這種情況下,你能改變指標的指向)。如果總是指向乙個物件並且一旦指向乙個物件後就不會改變指向,那麼你應該使用引用。 

還有一種情況,就是當你過載某個操作符時,你應該使用引用。最普通的例子是操作符。這個操作符典型的用法是返回乙個目標物件,其能被賦值。 

vectorv(10); // 建立整形向量(vector),大小為10; 

// 向量是乙個在標準c庫中的乙個模板(見條款35) 

v[5] = 10; // 這個被賦值的目標物件就是操作符返回的值 

如果操作符返回乙個指標,那麼後乙個語句就得這樣寫: 

*v[5] = 10; 

但是這樣會使得v看上去象是乙個向量指標。因此你會選擇讓操作符返回乙個引用。(這有乙個有趣的例外,參見條款30) 

當你知道你必須指向乙個物件並且不想改變其指向時,或者在過載操作符並為防止不必要的語義誤解時,你不應該使用指標。而在除此之外的其他情況下,則應使用指標 

假設你有 

void func(int* p, int&r); 

int a = 1; 

int b = 1; 

func(&a,b); 

指標本身的值(位址值)是以pass by value進行的,你能改變位址值,但這並不會改變指標所指向的變數的值, 

p = someotherpointer; //a is still 1 

但能用指標來改變指標所指向的變數的值, 

*p = 123131; // a now is 123131 

但引用本身是以pass by reference進行的,改變其值即改變引用所對應的變數的值 

r = 1231; // b now is 1231

引用不是值,不會佔據儲存空間。所以引用只有宣告,沒有定義。而指標是乙個實體,要佔據儲存空間

引用總是指向乙個是例項化的物件

c 中指標和引用的物理實現是一回事,都是記憶體位址;兩者的區別是在編譯時編譯器無法對指標操作進行型別檢查,而對引用可以。這也是引用更安全的原因。

實際上「引用」可以做的任何事情「指標」也都能夠做,為什麼還要「引用」 

這東西? 

答案是「用適當的工具做恰如其分的工作」。 

指標能夠毫無約束地操作記憶體中的如何東西,儘管指標功能強大,但是非常危險。 

就象一把刀,它可以用來砍樹、裁紙、修指甲、理髮等等,誰敢這樣用? 

如果的確只需要借用一下某個物件的「別名」,那麼就用「引用」,而不要用「指標」, 

以免發生意外。比如說,某人需要乙份證明,本來在檔案上蓋上公章的印子就行了,如 

果把取公章的鑰匙交給他,那麼他就獲得了不該有的權利。 

指標引用和引用指標的區別

c 在傳參的過程中,指標 和指標引用 是有區別的。雖然,我們往往可以通過傳指標然後通過間址訪問來修改指標所指向物件的值,同樣,通過引用也可以直接修改物件的值。但是,當傳指標的時候有乙個問題,就是,我們雖然可以通過指標來修改指標所指向物件的值,但是我們沒有不能直接修改指標的內容 也就是指標變數存放的位...

傳指標和傳指標引用的區別 指標和引用的區別(本質)

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

傳指標和傳指標引用的區別 指標和引用的區別(本質)

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