指針與引用的區別以及引用的三種用法


1.指針與引用的區別:

指針是一塊內存的地址值,而引用是一塊內存的別名。

下面引自:http://www.cnblogs.com/kingln/articles/1129114.html

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

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

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

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

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

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

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

程序在編譯時分別將指針和引用添加到符號表上,符號表上記錄的是變量名及變量所對應地址。指針變量在符號表上對應的地址值為指針變量的地址值,而引用在符號表上對應的地址值為引用對象的地址值。符號表生成后就不會再改,因此指針可以改變其指向的對象(指針變量中的值可以改),而引用對象則不能修改。

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

★相同點:

●都是地址的概念;

指針指向一塊內存,它的內容是所指內存的地址;而引用則是某塊內存的別名。

★不同點:

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

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

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

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

●“sizeof 引用”得到的是所指向的變量(對象)的大小,而“sizeof 指針”得到的是指針本身的大小;

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

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

舉例(今晚糾結不已的例子):

1)

 1 #include<iostream>
 2 #include<cstdio>
 3 
 4 using namespace std;
 5 
 6 void j(int* a)
 7 {
 8     a=new int;
 9     *a=12;
10 }
11 
12 int main()
13 {
14     int* a;
15     printf("%d\n",a);
16     j(a);
17     printf("%d\n",a);
18     printf("%d\n",*a);
19     return 0;
20 }
View Code

2)

 1 #include<iostream>
 2 #include<cstdio>
 3 
 4 using namespace std;
 5 
 6 void j(int** a)
 7 {
 8     *a=new int;
 9     **a=12;
10 }
11 
12 int main()
13 {
14     int* a;
15     printf("%d\n",a);
16     j(&a);
17     printf("%d\n",a);
18     printf("%d\n",*a);
19     return 0;
20 }
View Code

 

 

2.引用的三種用法:

1)獨立引用

 1 #include<iostream>
 2 
 3 using namespace std;
 4 
 5 
 6 
 7 int main()
 8 {
 9     int a=3;
10     int& b=a;//b與a綁定在一起,同生共死
11     cout<<b<<" "<<a<<endl;
12     b=5;
13     cout<<b<<" "<<a<<endl;
14     return 0;
15 }
View Code

2)作為函數參數

 1 #include<iostream>
 2 
 3 using namespace std;
 4 
 5 void f(int& b)//b在這里與實參a無異
 6 {
 7     b++;
 8 }
 9 
10 int main()
11 {
12     int a=3;
13     f(a);//a受函數體內部影響
14     cout<<a<<endl;
15     return 0;
16 }
View Code

3)作為函數的返回

 1 #include<iostream>
 2 
 3 using namespace std;
 4 
 5 int& f(int* p)
 6 {
 7     (*p)++;
 8     return *p;
 9 }
10 
11 int main()
12 {
13     int a=3,b;
14     b=f(&a)*5;
15     f(&a)+=10;
16     cout<<b<<" "<<a<<endl;//輸出20與15
17     return 0;
18 }
19 //f函數返回一個(*p)的引用,即a的引用。此引用可作為左值進行運算。
View Code

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM