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引用可以使得变量具有只读属性
- 引用的最终本质为为指针
- 引用可以尽可能的避开内存错误
C++中引用的本质分析的更多相关文章
- C++中引用的本质
一般的教材上讲到引用时,都是说"引用是对象的一个别名".我认为这种定义是不清晰的,不利于初学者理解引用.至少我自己曾经被这个定义困扰了一段时间.到底什么是"别名" ...
- C++中引用的本质是什么?
一般的教材上讲到引用时,都是说“引用是对象的一个别名”.我认为这种定义是不清晰的,不利于初学者理解引用.至少我自己曾经被这个定义困扰了一段时间.到底什么是“别名”? 实际上,引用的实质是位于xxxxx ...
- c语言中函数调用的本质从汇编角度分析
今天下午写篇博客吧,分析分析c语言中函数调用的本质,首先我们知道c语言中函数的本质就是一段代码,但是给这段代码起了一个名字,这个名字就是他的的这段代码的开始地址 这也是函数名的本质,其实也就是汇编中的 ...
- iOS中引用计数内存管理机制分析
在 iOS 中引用计数是内存的管理方式,虽然在 iOS5 版本中,已经支持了自动引用计数管理模式,但理解它的运行方式有助于我们了解程序的运行原理,有助于 debug 程序. 操作系统的内存管理分成堆和 ...
- flask中路由的本质源码分析
flask中url的本质: 吧url和视图函数封装到一个Rule对象里面去了,并且吧这个对象添加到url_map中 Rule={"url":'/index','method':'i ...
- Android XML中引用自定义内部类view的四个why
今天碰到了在XML中应用以内部类形式定义的自定义view,结果遇到了一些坑.虽然通过看了一些前辈写的文章解决了这个问题,但是我看到的几篇都没有完整说清楚why,于是决定做这个总结. 使用自定义内部类v ...
- C++中引用和指针详解
先来分析指针这个东东: 从概念上讲,指针本质上就是存放变量地址的一个变量,在逻辑上是独立的,它可以被改变,包括其所指向的地址的改变和其指向的地址中所存放的数据的改变. 上面的图表示了程序运行时变量的值 ...
- C++解析(4):引用的本质
0.目录 1.引用的意义 2.特殊的引用 3.引用的本质 4.函数返回引用 5.小结 1.引用的意义 引用作为变量別名而存在,因此在一些场合可以代替指针 引用相对于指针来说具有更好的可读性和实用性 注 ...
- C++ 类的多态五(多态的语法本质分析)
//多态的语法本质分析 #include<iostream> using namespace std; /* 三种易混淆的多态场景 */ class Point{ public: Poin ...
随机推荐
- Web博文目录
前言 博客写的多了,自己翻起来也费劲,这里就进行一下整合. 以前设想自己做DBA,做运维,没想到最后还要走开发这条路,干一行就爱一行...学的扎实点,工作起来也会轻松.—— 送给奋斗的自己 1 Jav ...
- 表空间常用sql汇总
表空间碎片相关:select tablespace_name, round(sqrt(max(blocks) / sum(blocks)) * (100 / sqr ...
- nginx知识图谱
nginx启动起来后有两个进程,一个是主进程,一个是守护进程
- unity3d中设计模式的学习<一>:泛型单例
单例是游戏开发中比较常见的设计模式,虽然针对的功能不同,但是有一些功能还是共有的,代码也不少,如果能放在一个基类里面是最好不过了,但是单例里需要有个instance功能来返回当前对象,所以这个功能必须 ...
- 全新Chrome Devtool Performance使用指南
运行时性能表现(runtime performance)指的是当你的页面在浏览器运行时的性能表现,而不是在下载页面的时候的表现.这篇指南将会告诉你怎么用Chrome DevToos Performan ...
- define常量
看手册说define定义的常量只允许: 仅允许标量和 null.标量的类型是 integer, float,string 或者 boolean. 也能够定义常量值的类型为 resource ,但并不推 ...
- BIND简易教程(1):安装及基本配置
首先,为什么说是简易教程呢?因为BIND的功能实在太多,全写出来的话要连载好久,我觉得我没有那么多精力去写:而我了解的仅仅是有限的一点点,不敢造次.百度上的文章也是一抓一大把呐!所以,教点基本使用方法 ...
- Map使用方法
转:https://www.cnblogs.com/lzq198754/p/5780165.html Java map 详解 - 用法.遍历.排序.常用API等 概要: java.util 中的集合类 ...
- 关于Queue的相关问题
在多线程中使用Queue,发现总是有莫名的问题, 经折腾好久之后发现是因为没有加锁! 以下测试代码中, 如果不加锁, 添加 100W对象, 可能只会成功50W, 然后并不会产生异常! );//(如果初 ...
- spring注入bean的三种方法
在Spring的世界中, 我们通常会利用bean config file 或者 annotation注解方式来配置bean. 在第一种利用bean config file(spring xml)方式中 ...