当前位置: 首页 > news >正文

C++入门(2)--引用

6.引用

引用不是新定义一个变量,而是给已存在变量取了一个别名,编译器不会为引用变量开辟内存空间,它和它引用的变量共用同一块内存空间。

类型& 引用变量名(对象名) = 引用实体;

//引用
//类型& 引用变量名(对象名) = 引用实体;
int main()
{int a = 0;int& b = a;int& c = b;int& d = a;b++;d++;cout << a << endl;int x =11;d = x;cout << a << endl;return 0;
}

注:引用类型必须和引用实体时同种类型

6.1 引用特性

1. 引用在定义时必须初始化

2. 一个变量可以有多个引用

3. 引用一旦引用一个实体,再不能引用其他实体

void TestRef()
{int a = 10;// int& ra; // 该条语句编译时会出错int& ra = a;int& rra = a;printf("%p %p %p\n", &a, &ra, &rra);
}

6.2 常引用

引用过程中,权限不能放大,但权限可以平移或者缩小

int main()
{//不可以,引用过程中,权限不能放大const int a = 10;//int& ra = a; // 该语句编译时会出错,a为常量const int& ra = a;// 可以,c拷贝给d,没有放大权限,因为d的改变不影响cconst int c = 0;int d = c;// 可以// 引用过程中,权限可以平移或者缩小int x = 0;int& y = x;const int& z = x;++x;//z的值也会变++z;//errordouble d = 12.34;int i=d;//会产生int临时变量,临时变量具有常性//int& ri = d; // 该语句编译时会出错,类型不同const int& ri = d;return 0;
}


int func1()
{static int x = 0;return x;
}int& func2()
{static int x = 0;return x;
}int main()
{// int& ret1 = func1();  // 权限放大//const int& ret1 = func1(); // 权限平移// int ret1 = func1();  // 拷贝int& ret2 = func2();        // 权限平移const int& rret2 = func2();  // 权限缩小return 0;
}

6.3 使用场景

1.做参数

引用参数(输出型参数);

引用参数(减少拷贝提高效率)(大对象/深拷贝类对象)

void Swap(int& a, int& b)
{int tmp = a;a = b;b = tmp;
}void Swap(int*& a, int*& b)
{int* tmp=a;a = b;b = tmp;
}int main()
{int x = 0, y = 1;Swap(x, y);cout << x << " " << y << endl;int* px = &x, * py = &y;cout << px << " " << py << endl;Swap(px, py);cout << px << " " << py << endl;cout << x << " " << y << endl;return 0;
}

2.做返回值

引用做返回值(减少拷贝提高效率)(大对象/深拷贝类对象)

引用做返回值 修改返回值+获取返回值

获取返回值
传值返回
int Count()
{static int n = 0;n++;return n;
}int main()
{int ret = Count();cout << ret << endl;return 0;
}传引用返回
int& Count()
{static int n = 0;n++;return n;
}int main()
{int ret = Count();cout << ret << endl;return 0;
}int& Count()
{int n=0;n++;return n;
}int main()
{int ret=Count();cout<<ret<<endl;return 0;
}
这里打印ret的值是不确定的
如果Count函数结束,栈帧销毁,没有清理栈帧,那么ret的结果侥幸是正确的
如果Count函数结束,栈帧销毁,清理栈帧,那么ret的结果是随机值//修改返回值
void SLModify( SeqList* ps, int pos, int x)
{assert(pos >= 0 && pos < 100);ps->a[pos] = x;
}int SLGet(SeqList* ps, int pos)
{assert(pos >= 0 && pos < 100);return ps->a[pos];
}//读写功能都有
int& SLAt(SeqList* ps, int pos)
{assert(pos >= 0 && pos < 100);return ps->a[pos];
}int main()
{SeqList s;SLModify(&s, 0, 1);cout << SLGet(&s, 0) << endl;//对第0个位置的值+5int ret1 = SLGet(&s, 0);SLModify(&s, 0, ret1 + 5);cout << SLGet(&s, 0) << endl;SLAt(&s, 0) = 2;cout << SLGet(&s, 0) << endl;SLAt(&s, 0) += 5;cout << SLGet(&s, 0) << endl;return 0;
}

总结:

1.基本任何场景都可以用引用传参;

2.谨慎使用引用做返回值,除了函数作用域,对象不在了,就不能用引用返回,还在就可以用引用返回

6.4 引用和指针的区别

int main()
{int a = 10;// 语法层面:不开空间,是对a取别名int& ra = a;ra = 20;// 语法层面:开空间,存储a的地址int* pa = &a;*pa = 30;return 0;
}

引用和指针的不同点:

1. 引用概念上定义一个变量的别名,指针存储一个变量地址。

2. 引用在定义时必须初始化,指针没有要求

3. 引用在初始化时引用一个实体后,就不能再引用其他实体,而指针可以在任何时候指向任何一个同类型实体

4. 没有NULL引用,但有NULL指针

5. 在sizeof中含义不同:引用结果为引用类型的大小,但指针始终是地址空间所占字节个数(32位平台下占4个字节)

6. 引用自加即引用的实体增加1,指针自加即指针向后偏移一个类型的大小

7. 有多级指针,但是没有多级引用

8. 访问实体方式不同,指针需要显式解引用,引用编译器自己处理

9. 引用比指针使用起来相对更安全


http://www.mrgr.cn/news/38252.html

相关文章:

  • FreeRTOS 内存管理源码解析
  • 使用VBA快速生成Excel工作表非连续列图片快照
  • 网页篡改防御方法
  • element-plus中el-table固定列fixed失效问题
  • 在LLMs模型中发现人类的记忆特征
  • nodejs 016: javascript语法——解构赋值({ a, b, c } = {})=>{console.log(“Hello“);}
  • linux编辑文件保存退出的实操讲解
  • 基于SpringCloud的微服务架构下安全开发运维准则
  • 微服务Redis解析部署使用全流程
  • Meta Sapiens 人体AI模型
  • 【设计模式】精通高级行为模式:揭秘状态、中介者及其它模式的深层机制
  • 【宝藏妙招,轻松拿捏!】如何防止U盘资料被复制?U盘文件防拷贝的五种措施!
  • Sharding-JDBC笔记03-分库分表代码示例
  • 【无人机设计与控制】基于蚁群算法的三维无人机航迹规划+无人机路径规划
  • 要想不被控制,就的金钱来买账,老板不是每个人能当的!
  • 2024版最新Wireshark安装使用教程(非常详细)零基础入门到精通,收藏这一篇就够了_wireshark 4.4.0安装要求
  • 【笔记】X射线的衍射方向
  • 98问答网是一个怎样的平台?它主要提供哪些服务?
  • 反转整数。
  • VS开发C++项目常用基础属性配置