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 ...
随机推荐
- Oracle日常性能查看
判断回滚段竞争的SQL语句:(当Ratio大于2时存在回滚段竞争,需要增加更多的回滚段)select rn.name, rs.GETS, rs.WAITS, (rs.WAITS / rs.GETS) ...
- 关于cmake输出动态链接库名字的问题
使用cmake进行项目编译管理时,我们经常使用 add_library(foo SHARED foo.cpp) 这样的话,输出时,如果在win下面会得到foo.dll,linux下面会得到libfoo ...
- Android Studio 独立引入(非友盟)微博分享和回调时问题
最近同事在做一个小项目时,由于产品的要求,Wap页面的分享规定不能使用友盟的社会化组件.他则不得不手动一个一个渠道的引入分享,好在渠道不多就三个,但是第一微博分享引入的时候问题就出现了. 问题一:li ...
- hdu-2837 Calculation---指数循环节
题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=2837 题目大意: 已知f(0) = 1,0^0 =1,[注意,0的其他任意次方为0,虽然题没有直接给 ...
- ctrl + alt + o 快速删除掉没有使用的 import
ctrl + alt + o 优化导入,可以快速删除掉没有使用的 import
- sort给文件按照大小排序
ls -l|sort -n -k5 -n 表示以数值排序-k5 表示以第几列排序还可以用 -t参数指定行内容的分隔符 参考链接:http://www.cnblogs.com/myd620/p/6002 ...
- SpringMVC WEB应用上传照片的实现
使用是SpringMVC+Hibernate搭建的WEB应用,使用jsp写的前端页面. 如何将文件上传到服务器呢?我这里使用的是Multipart的形式将文件上传. 这里有两大步:一是配置multip ...
- 理解JavaScript的this对象
1.概述 this对象是在运行时基于函数的执行环境绑定的,this总是返回一个对象,简单说,就是返回属性或方法"当前"所在的对象.在全局函数中,this等于window,而当函数作 ...
- canvas制作随机验证码
看到人家彩色背景的验证码想测试一下: 创建html代码: <canvas id="myCanvas" width="200" height="1 ...
- 用python实现ping
#!/usr/bin/env python #coding=utf-8 import os import argparse import socket import struct import sel ...