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:

 const int *cptr;
*cptr = ;//error,不允许通过解引用修改指针所指向的对象的值 const double dv = ;
double *dptr = &dv;//error,不能使用const类型的地址赋给非const类型的指针 double dv2 = ;
const double *dptr2 = &dv2;//非const类型的地址可以赋给const类型的指针 const int intValue = ;
void *vptr = &intValue;//error
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:

     int v1 = ;
int &rv1 = v1; rv1 = ;
cout<<"rv1 = "<<rv1<<" v1 = "<<v1<<endl;//rv1 = 12 v1 = 12 //int &rv2 = 10;//error,不能直接使用右值初始化引用
double dValue = 10.0; //int &rv3 = dValue;//error,只能用与引用同类型的对象初始化引用 const int &rv4 = dValue;//ok const int &rv5 = ;//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;
}

后续更新中......

上一篇:[Chapter 3 Process]Practice 3.1 相关知识:进程创建、fork函数


下一篇:[Chapter 3 Process]Practice 3.9 Describe the actions token by a kernel to content-switch between processes.