IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> C++知识库 -> C++之引用 -> 正文阅读

[C++知识库]C++之引用

目录

一、引用概念

二、引用特性

三、常引用(有一点点像常变量)

四、使用场景

1、做参数

2、做返回值

六、传值、传引用效率比较

七、引用和指针的区别

八、引用和指针的不同点:


一、引用概念

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

比如:李逵,在家称为"铁牛",江湖上人称"黑旋风"

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

void TestRef()

{

int a = 10;

int& ra = a;//<====定义引用类型

int& rra=a;

int& rrra=rra;

printf("%p\n", &a);

printf("%p\n", &ra);

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

二、引用特性

  1. 引用在定义时必须初始化// int& ra = a;初始化表明引用了a
  2. 一个变量可以有多个引用
  3. 引用一旦引用一个实体,再不能引用其他实体

void TestRef()

{

??? ??? int a = 10;

?????? int b=20;

??? ??? // int& ra;// 该条语句编译时会出错

?????? int& ra = a;

?????? int& rra = a;

?????? ra=b;//这里是把b的值赋值给了ra

?????? printf("%p %p %p\n", &a, &ra, &rra);

???

}

三、常引用(有一点点像常变量)

void TestConstRef()

{?? //1?????????????????

??? const int a = 10;(为左值)

??? //int& ra = a;??? // ra的类型是int,编译不通过因为a的类型为const int,是只读,ra的类型为int,也就是可读可写的,这是不允许的 (两者类型都为int)

??? const int& ra = a;//这样引用就可以了

??? //2

??? int c=1;

??? int &e=c;

??? const int&f=c;//这里是可以的,f只是读

//总结:引用取别名时,变量访问的权限可以缩小,不能放大

??? //3

??? // int& b = 10; // 该语句编译时会出错,b为常量

??? const int& b = 10;

??? //4

??? double d = 12.34;

??? //int& rd = d; // 该语句编译时会出错,类型不同(更确切的说是权限放大了)

??? const int& rd = d;

??? //补充1

??? const int ar=10;

??? int br=ar;//不是隐式类型转换,都为int,只是简单的赋值,br的变化不会影响ar

??? //补充2

??? const int * cp1=&a;

??? //int *p1=cp1;// 该语句编译时会出错,权限放大了

??? //补充3

??? int * cp1=&a;//a不为常变量

??? const int *p1=cp1;// 权限缩小。可以

}

四、使用场景

1、做参数

#define _CRT_SECURE_NO_WARNINGS 1

#include<iostream>

//指针

void swap_c(int*p, int*q)

{

??? int tmp = *p;

??? *p = *q;

??? *q = tmp;

}

//引用

void swap_cpp(int &r1, int&r2)

{

??? int tmp = r1;

??? r1 = r2;

??? r2 = tmp;

}

int main()

{

??? int a = 10;

??? int b = 23;

??? swap_c(&a, &b);

??? swap_cpp(a, b);

??? return 0;

}

2、做返回值

#define _CRT_SECURE_NO_WARNINGS 1

#include<iostream>

int Count1()

{

??? static int n = 0;

??? n++;

??? return n;//返回的是临时变量tmp,其中int tmp=n,会多创一个空间给tmp?

}

int& Count2()

{

??? static int n = 0;

??? n++;

??? return n;//返回的是tmptmpn的引用其中int& tmp=n ,因为n出了count2函数也不会被销毁

//则可以使用引用返回??

}

int main()

{

??? const int&r1 = Count1();//Count1()返回的tmp是临时变量,具有常属性,r1引用的是该临时变量

??? int&r2 = Count2();//Count2()返回的是tmptmpn的引用,因此r2也是n的引用,没有开辟空间给tmp,tmpn的引用

??? return 0;

}

#define _CRT_SECURE_NO_WARNINGS 1

#include<iostream>

using namespace std;

int& Add(int a, int b)

{

??? int c = a + b;

??? /*static int c = 0;

??? ?c = a + b;;*/

??? return c;

}

int main()

{

??? int& ret = Add(1, 2);//ret指向了c的空间,此时c已经被销毁了

??? Add(3, 4);//返回的c和第一次调用的c的空间恰好是同一个内存空间,因此ret的值变了

??? //release下会被优化,第二次调用创建的c和第一次的c不是同一个内存空间,因此ret的值未被修改

??? //这样引用局部变量是不安全的,因为别人可以通过其它途径修改ret的值

??? cout << "Add(1, 2) is :" << ret << endl;//ret7

??? return 0;

}

?注意:如果函数返回时,出了函数作用域,如果返回对象还未还给系统,则可以使用引用返回,如果已经还给系统了,则必须使用传值返回。

六、传值、传引用效率比较

以值作为参数或者返回值类型,在传参和返回期间,函数不会直接传递实参或者将变量本身直接返回,而是 传递实参或者返回变量的一份临时的拷贝,因此用值作为参数或者返回值类型,效率是非常低下的,尤其是? 当参数或者返回值类型非常大时,效率就更低。

七、值和引用的作为返回值类型的性能比较

#define _CRT_SECURE_NO_WARNINGS 1

#include <time.h>

#include<iostream>

using namespace std;

struct A

{

??? int a[10000];

};

A a;

// 值返回

A TestFunc1()

{

??? return a;

}

// 引用返回

A& TestFunc2()

{

??? return a;

}

void TestReturnByRefOrValue()

{

??? // 以值作为函数的返回值类型

??? size_t begin1 = clock();

??? for (size_t i = 0; i < 1000000; ++i)

?????? TestFunc1();

??? size_t end1 = clock();

??? // 以引用作为函数的返回值类型

??? size_t begin2 = clock();

??? for (size_t i = 0; i < 1000000; ++i)

?????? TestFunc2();

??? size_t end2 = clock();

??? // 计算两个函数运算完成之后的时间

??? cout << "TestFunc1 time:" << end1 - begin1 << endl;

??? cout << "TestFunc2 time:" << end2 - begin2 << endl;

}

int main()

{

??? TestReturnByRefOrValue();

??? return 0;

}

通过上述代码的比较,发现传值和指针在作为传参以及返回值类型上效率相差很大

使用引用便可以减少拷贝提高效率(当返回值足够大或者调用函数返回次数足够多这效率的提升的效果非常明显)

七、引用和指针的区别

语法概念上引用就是一个别名,没有独立空间,和其引用实体共用同一块空间。

底层实现上实际是有空间的,因为引用是按照指针方式来实现

int main()

{

??? int a = 10; int& ra = a;

??? cout << "&a = " << &a << endl;

??? cout << "&ra = " << &ra << endl;

??? return 0;

}

int main()

{

??? int a = 10;

??? int& ra = a;

??? ra = 20;

??? int* pa = &a;

??? *pa = 20;

??? return 0;

}

?我们来看下引用和指针的汇编代码对比:

?

八、引用和指针的不同点:

  1. 引用在定义时必须初始化,指针没有要求
  2. 引用在初始化时引用一个实体后,就不能再引用其他实体,而指针可以在任何时候指向任何一个同类型 实体
  3. 没有NULL引用,但有NULL指针
  4. 在sizeof中含义不同:引用结果为引用类型的大小,但指针始终是地址空间所占字节个数(32位平台下占4个字节)
  5. 引用自加即引用的实体增加1,指针自加即指针向后偏移一个类型的大小
  6. 有多级指针,但是没有多级引用
  7. 访问实体方式不同,指针需要显式解引用,引用编译器自己处理
  8. 引用比指针使用起来相对更安全
  C++知识库 最新文章
【C++】友元、嵌套类、异常、RTTI、类型转换
通讯录的思路与实现(C语言)
C++PrimerPlus 第七章 函数-C++的编程模块(
Problem C: 算法9-9~9-12:平衡二叉树的基本
MSVC C++ UTF-8编程
C++进阶 多态原理
简单string类c++实现
我的年度总结
【C语言】以深厚地基筑伟岸高楼-基础篇(六
c语言常见错误合集
上一篇文章      下一篇文章      查看所有文章
加:2021-09-22 14:29:00  更:2021-09-22 14:30:22 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/23 23:15:24-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码