C++中引用的本质分析

引用的意义

  • 引用作为变量别名而存在,因此在一些场合可以代替指针
  • 引用相对于指针来说具有更好的可读性和实用性
  • swap函数的实现对比:

void swap(int* a, int* b)

{

int t = *a;

*a = *b;

*b = t;

}

void swap_yinyong(int& a,int& b)

{

int t = a;

a = b;

b = t;

}

int main()

{

int a = 1;

int b = 2;

printf("a = %d,b = %d\n",a,b);

swap(&a,&b);

printf("指针版本:a = %d, b = %d\n",a,b);

swap_yinyong(a,b);

printf("引用版本:a = %d, b = %d\n",a,b);

return 0;

}

特殊的引用:const引用

  • 在C++中可以声明const引用

Const Type& name = val;

Const 引用让变量拥有只读属性

int main()

{

int a = 1;

const int& b = a;

int *p = (int*)&b;

//b = 5; 编译报错,显示不能对只读变量赋值

*p = 5; //要想改变只读变量,可以对他的指针进行操作

printf("a = %d\n",a);

return 0;

}

  • 当使用常量对const引用进行初始化时,C++编译器会为常量分配空间并将引用名作为这段空间的别名

int main()

{

const int& b = 1;

int *p = (int*)&b;

*p = 5; //要想改变只读变量,可以对他的指针进行操作

printf("a = %d\n",b);

return 0;

}

使用常量对const引用初始化后将生成一个只读变量

引用在C++中的内部实现是一个指针常量

  • 因此引用占用的内存空间与指针相同(一般是四个字节)
  • 从使用的角度,引用只是一个别名,C++为了实用性而隐藏了引用的存储空间这一细节

struct Tref

{

char& c;

};

int main()

{

char c = ‘c‘;

char& rc = c;

Tref ref = { c };

printf("sizeof(char&)%d\n",sizeof(char&));

printf("sizeof(rc)%d\n",sizeof(rc));

printf("sizeof(ref.c)%d\n",sizeof(ref.c));

printf("sizeof(Tref)%d\n",sizeof(Tref));

return 0;

}

struct Tref

{

char* before;

char& ref;

char* after;

};

int main()

{

char a = ‘a‘;

char b = ‘b‘;

char c = ‘c‘;

Tref r = {&a,b,&c};

printf("sizeof(r)%d\n",sizeof(r));

printf("sizeof(Tref.before)%d\n",sizeof(r.before));

printf("sizeof(Tref.after)%d\n",sizeof(r.after));

printf("&r.before=%p\n",&r.before);

printf("&r.after= %p\n",&r.after);

return 0;

}

引用的意义

  • C++中的引用大多数情况下能够代替指针
  • 功能性:可以满足多数需要使用指针的场合
  • 安全性:可以避开由于指针操作不当而带来的内存错误
  • 操作性:简单易用,又不失功能强大

小结:

  • 引用作为变量别名而存在旨在代替指针
  • const引用可以使得变量具有只读属性
  • 引用的最终本质为为指针
  • 引用可以尽可能的避开内存错误

原文地址:https://www.cnblogs.com/ryluo/p/10037288.html

时间: 2024-08-06 20:26:51

C++中引用的本质分析的相关文章

C++中引用的本质

一般的教材上讲到引用时,都是说"引用是对象的一个别名".我认为这种定义是不清晰的,不利于初学者理解引用.至少我自己曾经被这个定义困扰了一段时间.到底什么是"别名"? 实际上,引用的实质是位于xxxxxx地址上的一个xxxx类型的对象.比如教科书上常用的例子: int a = 5:           //不妨假设编译器将a分配到0x400000 int &b = a: 这里面b的准确意义就是,放在0x400000地址上的一个int类型对象.这里面包括了两重含

第5课 引用的本质分析

引用作为变量别名而存在,因此在一些场合可以代替指针 引用相对于指针来说具有更好的可读性和实用性 swap函数的实现对比如下: 注意: 函数中的引用形参不需要进行初始化. 示例程序如下: 形参没有初始化,而是在第15行调用的时候对引用形参进行初始化. const引用: 当使用常量对const引用进行初始化时,C++编译器会为常量值分配空间,并将引用名作为这段空间的别名. 例: 结论: 使用常量对const引用初始化后将生成一个只读变量.普通引用不能使用常量值进行初始化,但是const引用可以. 我

C++中引用和匿名对象的理解和本质剖析

引用是什么? 引用其实就是给变量起的一个别名,使用这个别名跟使用变量名没有区别. 那什么又是变量名呢? 变量名实质上是一段连续存储空间的别名,是一个标号(门牌号),编译器通过变量来申请并命名内存空间,程序员可以通过变量的名字可以使用存储空间. 也可以这样理解,变量名是逻辑概念,变量是物理层面,变量含数据类型和数据值,数据类型决定内存的分配,编译器将变量名和变量对应的内存联系起来,使程序员可以通过变量名来操作内存. 引用怎么用?    语法:Type& name = var; 规则:1.普通引用在

c语言中函数调用的本质从汇编角度分析

今天下午写篇博客吧,分析分析c语言中函数调用的本质,首先我们知道c语言中函数的本质就是一段代码,但是给这段代码起了一个名字,这个名字就是他的的这段代码的开始地址 这也是函数名的本质,其实也就是汇编中的标号.下面我们会接触到一些东西 比如 eip 就是我们常常说的程序计数器,还有ebp和esp (这里是俩个指针,记得我们以前学8086也就一个sp堆栈指针)分别为EBP是指向栈底的指针,在过程调用中不变,又称为帧指针.ESP指向栈顶,程序执行时移动,ESP减小分配空间,ESP增大释放空间,ESP又称

iOS中引用计数内存管理机制分析

在 iOS 中引用计数是内存的管理方式,尽管在 iOS5 版本号中.已经支持了自己主动引用计数管理模式.但理解它的执行方式有助于我们了解程序的执行原理.有助于 debug 程序. 操作系统的内存管理分成堆和栈. 在堆中分配的内存,都试用引用计数模式:在栈中则不是. NSString 定义的对象是保存在栈中,所以它没有引用计算.看一些书上说它的引用计算会是 fffffffff 最大整数.測试的结果显示它是- 1. 对该对象进行 retain 操作.不好改变它的 retainCount 值. Mut

iOS中引用计数内存管理机制分析总结(NSString引用计数为-1的情况)

在 iOS 中引用计数是内存的管理方式,虽然在 iOS5 版本中,已经支持了自动引用计数管理模式,但理解它的运行方式有助于我们了解程序的运行原理,有助于 debug 程序.   操作系统的内存管理分成堆和栈. 在堆中分配的内存,都适用引用计数模式:在栈中则不是. NSString 定义的对象是保存在栈中,所以它没有引用计数,不是通过引用计数对内存进行管理的.之前我在一篇博客上看,说常量的引用计数会是一个很大的整数,测试的结果显示它是-1. 对该对象进行 retain 操作,不好改变它的 reta

Android XML中引用自定义内部类view的四个why

今天碰到了在XML中应用以内部类形式定义的自定义view,结果遇到了一些坑.虽然通过看了一些前辈写的文章解决了这个问题,但是我看到的几篇都没有完整说清楚why,于是决定做这个总结. 使用自定义内部类view的规则 本文主要是总结why,所以先把XML布局文件中引用内部类的自定义view的做法摆出来,有四点: 自定义的类必须是静态类: 使用view作为XML文件中的tag,注意,v是小写字母,小写字母v,小写字母v: 添加class属性,注意,没有带android:命名空间的,表明该自定义view

C++中引用传递和指针传递函数参数的详解

先来分析指针这个东东: 从概念上讲,指针本质上就是存放变量地址的一个变量,在逻辑上是独立的,它可以被改变,包括其所指向的地址的改变和其指向的地址中所存放的数据的改变. 上面的图表示了程序运行时变量的值和地址,这时的内存长什么样子呢? 注意指针是一个变量,它当然有内存空间,里面存的就是一个地址,通过这个地址我们就能找到它所指向的对象. 说明:上图中两个字母p和n在最左边,代表什么?后面在介绍程序的编译过程中用到,先卖个官司.如果下面的写的东西你看不懂,没关系,往下看,我不相信你看完最后的编译原理的

C++中几个值得分析的小问题(2)

下面有3个小问题,作为C++ Beginner你一定要知道错在哪里了. 1.派生类到基类的引用或指针转换一定“完美”存在? 一般情况,你很可能会认为:派生类对象的引用或指针转换为基类对象的引用或指针是一件很正常的事.那要是不一般情况呢?请看下面这个例子: class Person { public: Person(const string& str = "Normal Person") : ID(str) {} string ID; //作为一般的人身份是“普通人”,作为学生身