C++学习基础二——指针与引用的区别
一、指针:
(1)如果对指针进行解引用操作 赋值,改变的是指针所指向对象的值;
(2)如果不对指针进行解引用操作 赋值,则改变的是指针本身的值;
(3)指向const对象的指针:也叫指针常量,表示指针所指向的对象是const类型的,不允许通过解引用修改其所指向的值。如代码片段2所示。
需要注意的是:指向const对象的指针不能通过解引用修改其所指向的对象的值;不能用const类型的地址初始化普通的,非const类型的指针;可以使用非const的地址初始化const类型的指针;可以用const类型的地址初始化const void *指针,而不能初始化void *指针。
(4)const指针:表示声明的指针是const类型的,不允许修改指针的值,但可以通过解引用修改指针所指向对象的值。跟const变量类似,const指针必须在声明的时候初始化。
例如:
代码片段1:
string s1("some value"); string *sp1 = &s1;//此时s1的值为“some value”,指针sp1指向s1 string s2("another"); string *sp2 = &s2;//此时s2的值为"another",指针sp2指向s2 *sp1 = "a new value";//此时s1的值为"a new value",指针sp1指向s1 sp1 = sp2;//此时指针sp1指向s2
代码片段2:
1 const int *cptr; 2 *cptr = 20;//error,不允许通过解引用修改指针所指向的对象的值 3 4 const double dv = 20; 5 double *dptr = &dv;//error,不能使用const类型的地址赋给非const类型的指针 6 7 double dv2 = 30; 8 const double *dptr2 = &dv2;//非const类型的地址可以赋给const类型的指针 9 10 const int intValue = 42; 11 void *vptr = &intValue;//error 12 const void *vptr2 = & intValue;//ok
13
14 const int *iptr;
15 int const *iptr2;// 14行和15行声明的指针是等价的,表示指针所指向的对象是const的,不能通过解引用修改,但指针本身的值是可以修改的
16 int *const *iptr3;//表示iptr3是const指针,指针的值不允许修改,但是可以通过解引用修改指针所指向对象的值
二、引用:
(1)定义引用时必须初始化。
(2)引用一经初始化,就始终指向特定的对象,如果给引用赋值,则修改的是引用所关联对象的值。
(3):非const类型的引用只能用该引用同类型的对象初始化,不能使用右值
(4):const类型的引用既可以使用右值初始化,也可以使用不同但相关类型的对象初始化
(5):const类型的引用一经初始化完成就不能修改引用的值
代码片段3:
int i1 = 1000,i2 = 2000; int *s1 = &i1, *s2 = &i2; s1 = s2; 结果:s1所指向的i1的值不变,赋值操作结束后,只是改变了指针s1本身的值,即s1指针指向另一个不同的对象。
代码片段4:
int &s1 = i1, &s2 = i2; s1 = s2; 结果:赋值操作修改了s1所关联对象i1的值,而不是引用本身的值。赋值结束后,引用s1和引用s2还是分别指向原来对象,只是此时两个对象的值相等。
代码片段5:
1 int v1 = 10; 2 int &rv1 = v1; 3 4 rv1 = 12; 5 cout<<"rv1 = "<<rv1<<" v1 = "<<v1<<endl;//rv1 = 12 v1 = 12 6 7 //int &rv2 = 10;//error,不能直接使用右值初始化引用 8 double dValue = 10.0; 9 10 //int &rv3 = dValue;//error,只能用与引用同类型的对象初始化引用 11 12 const int &rv4 = dValue;//ok 13 14 const int &rv5 = 10;//ok
给出比较全的代码:
#include <iostream> using namespace std; int main(){ float s1 = 100,s2 = 200; cout<<"原始两个值: "<<s1 << " "<<s2<<endl; float *sp1 = &s1, *sp2 = &s2; cout<<"第一次操作后的两个值: "<<s1 << " "<<s2<<endl; cout<<"两个指针的值 sp1 = "<<*sp1<<" sp2 = "<<*sp2<<endl; cout<<"-----------------------------"<<endl; //对指针进行解引用,改变了指针所指对象s1的值,由100变为10,对应上面指针理论(1) *sp1 = 10; cout<<"第二次操作后的两个值: "<<s1 << " "<<s2<<endl; cout<<"两个指针的值 sp1 = "<<*sp1<<" sp2 = "<<*sp2<<endl; cout<<"-----------------------------"<<endl; //不改变指针所指对象s1的值,改变了指针本身的值,此时指针sp1指向sp2,对应上面指针理论(2) sp1 = sp2; cout<<"第二次操作后的两个值: "<<s1 << " "<<s2<<endl; cout<<"两个指针的值 sp1 = "<<*sp1<<" sp2 = "<<*sp2<<endl; cout<<"-----------------------------"<<endl; //此时改变指针对象s2的值,同时指针sp2的值也改变 *sp1 = 15; cout<<"第三次操作后的两个值: "<<s1 << " "<<s2<<endl; cout<<"两个指针的值 sp1 = "<<*sp1<<" sp2 = "<<*sp2<<endl; cout<<"-----------------------------"<<endl; cout<<""<<endl; float &i1 = s1, &i2 = s2; cout<<"第四次操作后的两个值: "<<s1 << " "<<s2<<endl; cout<<"两个引用的值 s1 = "<<s1<<" s2 = "<<s2<<endl; cout<<"-----------------------------"<<endl; //对引用赋值改变的是引用所关联对象的值 i1 = 30; cout<<"第四次操作后的两个值: "<<s1 << " "<<s2<<endl; cout<<"两个引用的值 s1 = "<<s1<<" s2 = "<<s2<<endl; cout<<"-----------------------------"<<endl; //此时只是改变了i1引用所关联对象s1的值,只是引用i1和i2还是分别指向各自的对象,只是各自对象的值相等。 i1 = i2; cout<<"第五次操作后的两个值: "<<s1 << " "<<s2<<endl; cout<<"两个引用的值 s1 = "<<s1<<" s2 = "<<s2<<endl; cout<<"-----------------------------"<<endl; return 0; }
后续更新中......