引用及指针和引用的区别

article/2025/10/3 0:05:58

一.在C语言中,我们要给函数传参,有两种方法。

1.传值

void Swap(int a,int b)
{int tmp=a;a=b;b=tmp;
}
int main()
{int a=10;int b=20;Swap(a,b);return 0;
}

优点:形参不影响实参(保护实参),可读性强。
缺点:不能通过形参改变实参,效率低(需要用实参拷贝形参)。
2.传地址

void Swap(int* a,int* b)
{int tmp=*a;*a=*b;*b=*tmp;
}
int main()
{int a=10;int b=20;Swap(&a,&b);return 0;
}

优点:可以通过形参改变外部的实参 效率高(不会创建副本)。
缺点 : 不安全,可读性差。

二.引用

C++为我们提供了新的一种传值方法—>传引用.

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

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

int main()
{int a = 10;int b = 30;int& ra = a;//类型& 引用变量名(对象名)=引用实体;cout << &ra << " = " << &a << endl;ra = 20;//a将被修改为20return 0;
}

注意:引用类型必须和引用实体是同种类型的。

引用特性

  1. 引用在定义时必须初始化
  2. 一个变量可以有多个引用
  3. 引用一旦引用一个实体,再不能引用其他实体
 void TestRef()
{int a = 10;// int& ra; // 该条语句编译时会出错int& ra = a;int& rra = a;rra=20;//a,ra,rra都修改为20printf("%p %p %p\n", &a, &ra, &rra);
}

常引用

void TestConstRef()
{const int a = 10;//int& ra = a; // 该语句编译时会出错,a为常量const int& ra = a;// int& b = 10; // 该语句编译时会出错,b为常量const int& b = 10;double d = 12.34;//int& rd = d; // 该语句编译时会出错,类型不同const int& rd = d;//int和double可以进行隐式类型转换,编译器自己维护了一个空间,里面存的常量12,//我们不知道此空间的地址,也不知道此空间的名字,所以具有常属性,需要const类型的引用来引用。
}

引用的使用场景

1.可以做函数的形参

void Swap(int& a,int& b)
{int tmp=a;a=b;b=tmp;
}
int main()
{int a=10;int b=20;Swap(a,b);return 0;
}

可以做到通过形参修改实参的值,而且不易出错,效率也高。

2.可以作为返回值的类型:只要返回的变量生命周期比函数长就行。

int& Add(int left, int right)
{int ret= left + right;return ret;
}
int main()
{int a = 10;int b = 20;int& retVal=Add(a, b);Add(100, 200);return 0;
}

运行结果如下:
在这里插入图片描述
在这里插入图片描述
错误的原因:第一次调用Add函数时,ret的值被修改为30,被返回引用,此时ret是栈上的空间,只是因为调用函数被临时开辟出来的,所以调用完之后空间被回收,ret中存的是垃圾值。第二次调用Add函数时,又需要开辟函数运行的空间,OS就把上次的空间给了出来,ret的值被修改为300,所以retVal被改为300。

结论:返回类型为引用时,不能返回栈上的空间。只要返回的变量生命周期比函数长就行。

传值、传指针和传引用效率比较

#include <time.h>
struct A
{int a[10000];
};
void TestFunc1(A a)
{}
void TestFunc2(A* a)
{}
void TestFunc3(A& a)
{}
void TestRefAndValue()
{A a;// 以值作为函数参数size_t begin1 = clock();for (size_t i = 0; i < 10000; ++i)TestFunc1(a);size_t end1 = clock();// 以指针作为函数参数size_t begin2 = clock();for (size_t i = 0; i < 10000; ++i)TestFunc2(&a);size_t end2 = clock();// 以引用作为函数参数size_t begin3 = clock();for (size_t i = 0; i < 10000; ++i)TestFunc3(a);size_t end3 = clock();// 分别计算两个函数运行结束后的时间cout << "TestFunc1(int)-time:" << end1 - begin1 << endl;cout << "TestFunc2(int*)-time:" << end2 - begin2 << endl;cout << "TestFunc3(int&)-time:" << end3 - begin3 << endl;
}
// 运行多次,检测值,指针和引用在传参方面的效率区别
int main()
{for (int i = 0; i < 10; ++i){TestRefAndValue();}return 0;
}

在这里插入图片描述

结论:传值最慢(需要拷贝一份实参的副本),指针和引用效率差不多。

引用和指针的区别

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

int main()
{int a = 10;int& ra = a;cout<<"&a = "<<&a<<endl;cout<<"&ra = "<<&ra<<endl;return 0;
}

在这里插入图片描述

引用在底层就是按照指针类型来处理的。

//int a=10,b=20;
//int& ra = a;相当于int* const pa = &a;
//pa = &b;//错误
//const int& cra=a;相当于const int* const pa;

int main()
{int a = 10;int* pa = &a;int& ra = a;return 0;
}

在这里插入图片描述
反汇编代码是相同的。

引用和指针的不同点:

  1. 引用在定义时必须初始化,指针没有要求
  2. 引用在初始化时引用一个实体后,就不能再引用其他实体,而指针可以在任何时候指向任何一个同类型实体
  3. 没有NULL引用,但有NULL指针
  4. 在sizeof中含义不同:引用结果为引用类型的大小,但指针始终是地址空间所占字节个数(32位平台下占4个字节)
  5. 引用自加即引用的实体增加1,指针自加即指针向后偏移一个类型的大小
  6. 有多级指针,但是没有多级引用
  7. 访问实体方式不同,指针需要显式解引用,引用编译器自己处理
  8. 引用比指针使用起来相对更安全

右值引用

传送门:https://www.cnblogs.com/likaiming/p/9045642.html


http://chatgpt.dhexx.cn/article/HWbINorA.shtml

相关文章

C++ | 指针和引用的区别

01. C——指针和引用的区别 指针是一个存储变量地址的变量&#xff0c;指向内存的一个存储单元。 引用只是一个别名。 int a1; int *p&a;int a1; int &ba;使用sizeof看一个指针的大小是8&#xff0c;而引用则是被引用对象的大小。指针可以被初始化为NULL&#xff0c;…

C++—指针与引用的区别与联系

一、为什么要有引用&#xff08;C语言没有&#xff09; ▪ 因为引⽤和值有⼀样的语义&#xff0c;而指针不是 ▪ 不存在空引⽤&#xff0c;必须初始化&#xff1b;保证值不变&#xff0c;保证编译器更加安全 ▪ 加减号、赋值操作符&#xff0c;作⽤在引用上会触发对象的操作符重…

C++指针与引用的区别

指针和引用的区别 ①指针是一个变量&#xff0c;存储一个成员的地址&#xff1b;引用是一个常量&#xff08;指针常量&#xff09;&#xff0c;相当于一个成员的别名 ②指针声明和定义可以分开&#xff1b;引用声明时必须初始化 int* a;//指针声明 anew int(1);//指针定义 int…

【C++】---指针和引用的区别

指针和引用的区别 两者本质两者区别两者的相同点为什么传引用比传指针更安全两者做返回值效率的比较 两者本质 引用是别名&#xff0c;指针是地址、实体 两者区别 不同点分析1.初始化要求不同引用在创建的同时必须初始化&#xff0c;即引用到一个有效的对象&#xff0c;而指…

【C++】指针和引用的区别及指针传递和引用传递的区别

一、指针和引用的区别&#xff1a; 1.指针是一个变量&#xff0c;只不过这个变量存储的是一个地址&#xff0c;指向内存的一个存储单元&#xff1b;引用和原来的变量实质上是同一个东西&#xff0c;只不过是原变量的一个别名。 2.指针的值在初始化后可变&#xff0c;即指向其…

指针和引用的区别以及引用与指针基础

1引用&#xff1a; 引用&#xff08;reference&#xff09;为对象起了另外一个新的名字通过将声明符写成&d的形式来定义引用类型&#xff0c;其中d是声明的变量名&#xff1a; int ival 1024;int& refval ival;int& refval2;//报错&#xff1a;引用必须被初始化…

C++中指针和引用的区别(超详细)

指针和引用主要有以下区别&#xff1a; 引用必须被初始化&#xff0c;但是不分配存储空间。指针不声明时初始化&#xff0c;在初始化的时候需要分配存储空间。 引用初始化后不能被改变&#xff0c;指针可以改变所指的对象。 不存在指向空值的引用&#xff0c;但是存在指向空值的…

c++:指针和引用的区别

目录 前言&#xff1a; 1、引用概念上是定义一个变量的别名&#xff0c;而指针是存储一个变量的地址。 2、引用在定义时必须要初始化&#xff0c;但是指针没有要求。 3、引用在初始化时引用一个实体后&#xff0c;就不能再引用其他实体&#xff0c;因为其本质是一个指针常量…

C++中指针和引用的区别

1.指针和引用的定义和性质区别&#xff1a; &#xff08;1&#xff09;指针&#xff1a;指针是一个变量&#xff0c;只不过这个变量存储的是一个地址&#xff0c;指向内存的一个存储单元&#xff1b;而引用跟原来 的变量实质上是同一个东西&#xff0c;只不过是原变量的一个别…

C++中的指针与引用

写在前面 指针和引用形式上很好区别&#xff0c;但是他们似乎有相同的功能,都能够直接引用对象&#xff0c;对其进行直接的操作。但是什么时候使用指针&#xff1f;什么时候使用引用呢&#xff1f;这两者很容易混淆,在此我详细介绍一下指针和引用&#xff0c;力争将最真实的一…

引用和指针概念及区别

一、引用和指针 指针&#xff1a;指针是一个特殊的变量&#xff0c;它里面存储的的数值为内存里的一个地址&#xff0c;通过*访问内存地址所指向的值 引用&#xff1a;引用不是新定义一个变量&#xff0c;而是给已存在变量取了一个别名&#xff0c;编译器不会为引用变量开辟内…

01背包.动态规划.c语言实现

二维dp数组01背包 确定dp数组以及下标的含义 使用二维数组&#xff0c;即dp[i][j] 表示从下标为[0-i]的物品里任意取&#xff0c;放进容量为j的背包&#xff0c;价值总和最大是多少 2.递推公式&#xff1a;dp[i][j]dp[i-1][j-weight]value[i] dp[i][j]dp[i-1][j] 3.数组初始…

【动态规划】背包问题

注意&#xff1a;for j in range(1&#xff0c;m1):是枚举所有的情况&#xff0c;不用一一判断放入物品后背包容量减少后j的变化。为什么从1开始&#xff0c;因为0已经写出来了&#xff0c;即dp[i-1][j]dp[i-1][j-0*a[i-1]]0*v[i-1]。 01背包无价值 dp定义&#xff1a;前i项物…

动态规划——背包问题(01背包问题)

动态规划——背包问题&#xff08;01背包问题&#xff09;01背包问题(求最大价值)&#xff1a;问题优化01背包问题(求方案数)&#xff1a; 动态规划——背包问题&#xff08;01背包问题&#xff09; 01背包问题(求最大价值)&#xff1a; 有N件物品和一个最多能背重量为W的背包…

动态规划:关于01背包问题 I

动态规划&#xff1a;关于01背包问题&#xff0c;你该了解这些&#xff01; 对于面试的话&#xff0c;其实掌握01背包&#xff0c;和完全背包&#xff0c;就够用了&#xff0c;最多可以再来一个多重背包。 如果这几种背包&#xff0c;分不清&#xff0c;我这里画了一个图&…

c++ 动态规划-01背包

动态规划 - 01背包问题 1.使用递归遍历(穷举)求解&#xff1a; 01背包问题&#xff1a;给定 n 种物品和一个重量(容量)(限定条件)为 w 的背包,物品 i 的重量是 wi,其价值为 vi。(每种物品只有一个)问:如何选择装入背包的物品,使得装入背包中的物品的价值最大。 //VC6.0------…

动态规划——01背包问题(C++实现)

题目描述&#xff1a; 解题思路&#xff1a; 整体思路&#xff1a; 利用动态规划&#xff0c;其目的就是将原问题分解成几个子问题&#xff0c;通过求解简单的子问题&#xff0c;把原问题给解决&#xff0c;就比如斐波那契数列方程&#xff1a; f[i]f[i-1]f[i-2]; 动态规划的…

动态规划总结三01背包问题

01背包问题一般是利用动态规划进行解题的&#xff0c;这里通过leetcode1049来讲解01背包的解题思路以及如何对01背包应用题目转换和理清思路 01背包问题&#xff1a; 这里借用学习公众号代码随想录的一张图来说明背包问题的种类 对于面试的话&#xff0c;其实掌握01背包&…

dp 动态规划 01背包问题 Python

参考学习网址&#xff1a; https://www.bilibili.com/video/av33930433?fromsearch&seid10637513335818789097 https://www.cnblogs.com/anzhengyu/p/11408466.html 问题描述&#xff1a; 给定 n 件物品&#xff0c;物品的重量为 w[i]&#xff0c;物品的价值为 v[i]。现…

Python3使用动态规划处理01背包问题

文章目录 视频教程讲解题目介绍题解1&#xff1a;二维列表题解2&#xff1a;一维列表&#xff08;滚动数组&#xff09;延伸阅读 视频教程讲解 【Python算法系列】动态规划2-01背包问题&完全背包问题【Python算法实战】背包问题 题目介绍 原题链接&#xff1a;NC145 01背…