指針和引用在C++中很常用耀销,但是對于它們之間的區(qū)別很多人都不是太熟悉,下面來解釋下他們2者之間的區(qū)別和用法富腊。
1.指針和引用的定義和性質(zhì)區(qū)別:
(1)指針:指針是一個變量腺占,只不過這個變量存儲的是一個地址,指向內(nèi)存的一個存儲單元毒租;而引用跟原來的變量實(shí)質(zhì)上是同一個東西稚铣,只不過是原變量的一個別名而已。如:
int a=1;int *p=&a;
int a=1;int &b=a;
上面定義了一個整形變量和一個指針變量p墅垮,該指針變量指向a的存儲單元从媚,即p的值是a存儲單元的地址。
而下面2句定義了一個整形變量a和這個整形a的引用b噩凹,事實(shí)上a和b是同一個東西临扮,在內(nèi)存占有同一個存儲單元。
(2)可以有const指針剃允,但是沒有const引用沛简;
(3)指針可以有多級,但是引用只能是一級(int **p斥废;合法 而 int &&a是不合法的)
(4)指針的值可以為空椒楣,但是引用的值不能為NULL,并且引用在定義的時候必須初始化牡肉;
(5)指針的值在初始化后可以改變捧灰,即指向其它的存儲單元,而引用在進(jìn)行初始化后就不會再改變了统锤。
(6)"sizeof引用"得到的是所指向的變量(對象)的大小毛俏,而"sizeof指針"得到的是指針本身的大小饲窿;
(7)指針和引用的自增(++)運(yùn)算意義不一樣煌寇;
2.指針和引用作為函數(shù)參數(shù)進(jìn)行傳遞時的區(qū)別:
(1)指針作為參數(shù)進(jìn)行傳遞:
#include<iostream>
using namespace std;
void swap(int *a,int *b)
{
int temp=*a;
*a=*b;
*b=temp;
}
int main(void)
{
int a=1,b=2;
swap(&a,&b);
cout<<a<<" "<<b<<endl;
system("pause");
return 0;
}
結(jié)果為
2 1
用指針傳遞參數(shù),可以實(shí)現(xiàn)對實(shí)參進(jìn)行改變的目的逾雄,是因?yàn)閭鬟f過來的是實(shí)參的地址阀溶,因此使用*a實(shí)際上是取存儲實(shí)參的內(nèi)存單元里的數(shù)據(jù)腻脏,即是對實(shí)參進(jìn)行改變,因此可以達(dá)到目的银锻。
再看一個程序;
#include<iostream>
using namespace std;
void test(int *p)
{
int a=1;
p=&a;
cout<<p<<" "<<*p<<endl;
}
int main(void)
{
int *p=NULL;
test(p);
if(p==NULL)
cout<<"指針p為NULL"<<endl;
system("pause");
return 0;
}
運(yùn)行結(jié)果為:
0x7fff5fbff5e4 1
指針p為NULL
大家可能會感到奇怪永品,怎么回事,不是傳遞的是地址么击纬,怎么p回事NULL鼎姐?事實(shí)上,在main函數(shù)中聲明了一個指針p掉弛,并賦值為NULL症见,當(dāng)調(diào)用test函數(shù)時,事實(shí)上傳遞的也是地址殃饿,只不過傳遞的是指地址谋作。也就是說將指針作為參數(shù)進(jìn)行傳遞時,事實(shí)上也是值傳遞乎芳,只不過傳遞的是地址遵蚜。當(dāng)把指針作為參數(shù)進(jìn)行傳遞時,也是將實(shí)參的一個拷貝傳遞給形參奈惑,即上面程序main函數(shù)中的p和test函數(shù)中使用的p不是同一個變量吭净,存儲2個變量p的單元也不相同(只是2個p指向同一個存儲單元),那么在test函數(shù)中對p進(jìn)行修改肴甸,并不會影響到main函數(shù)中的p的值寂殉。
如果要想達(dá)到也同時修改的目的的話,就得使用引用了原在。
3.將引用作為函數(shù)的參數(shù)進(jìn)行傳遞:
在講引用作為函數(shù)參數(shù)進(jìn)行傳遞時友扰,實(shí)質(zhì)上傳遞的是實(shí)參本身,即傳遞進(jìn)來的不是實(shí)參的一個拷貝庶柿,因此對形參的修改其實(shí)是對實(shí)參的修改村怪,所以在用引用進(jìn)行參數(shù)傳遞時,不僅節(jié)約時間浮庐,而且可以節(jié)約空間甚负。
看下面這個程序:
#include<iostream>
using namespace std;
void test(int &a)
{
cout<<&a<<" "<<a<<endl;
}
int main(void)
{
int a=1;
cout<<&a<<" "<<a<<endl;
test(a);
system("pause");
return 0;
}
輸出結(jié)果為:
0x7fff5fbff5e4 1
0x7fff5fbff5e4 1
再看下這個程序:
這足以說明用引用進(jìn)行參數(shù)傳遞時,事實(shí)上傳遞的是實(shí)參本身审残,而不是拷貝梭域。
所以在上述要達(dá)到同時修改指針的目的的話,就得使用引用了维苔。
#include<iostream>
using namespace std;
void test(int *&p)
{
int a=1;
p=&a;
cout<<p<<" "<<*p<<endl;
}
int main(void)
{
int *p = NULL;
test(p);
if (p != NULL) {
cout << "p不為NULL "<<p;
}
system("pause");
return 0;
}
輸出結(jié)果為:
0x7fff5fbff5e4 1
p不為NULL 0x7fff5fbff5e4