【C++修行之道】(引用、函数提高)

目录

一、引用

1.1引用的基本使用

 1.2 引用注意事项

1.3 引用做函数参数

1.4 引用做函数返回值

1.5 引用的本质

1.6 常量引用

1.7引用和指针的区别

二、函数提高

2.1 函数默认参数

2.2函数占位参数

2.3 函数重载

2.4函数重载注意事项


一、引用

1.1引用的基本使用

作用: 给变量起别名

语法: 数据类型 &别名 = 原名

引用是别名,即为某个变量提供的另一个名字。一旦引用被初始化为一个对象,它就不能被指向另一个对象。引用没有自己的内存地址,它与所引用的对象共享同一块内存地址。

示例:

int main()
{//引用基本用法//数据类型	&别名 = 原名int a = 10;//创建引用int& b = a;cout << "a = " << a << endl;cout << "a = " << b << endl;b = 100;cout << "a = " << a << endl;cout << "b = " << b << endl;system("pause");return 0;
}

 

 1.2 引用注意事项

  • 引用必须初始化

  • 引用在初始化后,不可以改变

示例:

int main() {int a = 10;// 1.引用必须初始化// int &b;//错误,引用必须要初始化 int b = 20;// 2.引用在初始化后,不可以改变int& c = a; //一旦初始化后,就不可以更改c = b; //这是赋值操作,不是更改引用cout << "a = " << a << endl;cout << "b = " << b << endl;cout << "c = " << c << endl;system("pause");return 0;
}

1.3 引用做函数参数

作用:函数传参时,可以利用引用的技术让形参修饰实参

优点:可以简化指针修改实参

示例:

// 引用做函数参数
//1. 值传递
void mySwap01(int a, int b) {int temp = a;a = b;b = temp;
}
//2. 地址传递
void mySwap02(int* a, int* b) {int temp = *a;*a = *b;*b = temp;
}
//3. 引用传递
void mySwap03(int& a, int& b) {int temp = a;a = b;b = temp;
}
int main() {int a = 15;int b = 25;​//mySwap01(a, b);//值传递,形参不会修饰实参//cout << "a:" << a << " b:" << b << endl;//mySwap02(&a, &b);//地址传递,形参会修饰实参//cout << "a:" << a << " b:" << b << endl;​mySwap03( a , b );cout << "a:" << a << " b:" << b << endl;system("pause");return 0;
}

总结:通过引用参数产生的效果同按地址传递是一样的。引用的语法更清楚简单

1.4 引用做函数返回值

作用:引用是可以作为函数的返回值存在的

注意:不要返回局部变量引用

用法:函数调用作为左值

示例:

//引用做函数的返回值
//1.不要返回局部变量的引用
int& test01() {int a = 10; //局部变量存放在四区中的 栈区return a;
}int main() {//不能返回局部变量的引用int& ref = test01();cout << "ref = " << ref << endl;//如果第一次结果正确,是因为编译器做了保留cout << "ref = " << ref << endl;//第二次结果错误,因为a的内存已经释放system("pause");return 0;
}
//引用做函数的返回值
//2.函数调用可以作为左值(赋值符的左边是左值)
//返回静态变量引用
int& test02() {static int a = 20;//静态变量,存放在全局区,全局上的数据在程序结束后系统是否return a;
}int main() {//如果函数做左值,那么必须返回引用int& ref2 = test02();cout << "ref2 = " << ref2 << endl;cout << "ref2 = " << ref2 << endl;test02() = 1000;//如果函数的返回值是引用,这个函数调用可以作为左值cout << "ref2 = " << ref2 << endl;cout << "ref2 = " << ref2 << endl;system("pause");return 0;
}

1.5 引用的本质

本质:引用的本质在c++内部实现是一个指针常量.

讲解示例:

//发现是引用,转换为 int* const ref = &a;
void func(int& ref) {ref = 100; // ref是引用,转换为*ref = 100
}
int main() {int a = 10;//自动转换为 int* const ref = &a; 指针常量是指针指向不可改,也说明为什么引用不可更改int& ref = a;ref = 20; //内部发现ref是引用,自动帮我们转换为: *ref = 20;cout << "a:" << a << endl;cout << "ref:" << ref << endl;func(a);// 调用func函数,传递a的引用,此时a的值会被修改为100return 0;
}

引用的本质就是一个指针常量。

引用一旦初始化后,就不可以发生改变。

结论:C++推荐用引用技术,因为语法方便,引用本质是指针常量,但是所有的指针操作编译器都帮我们做了

1.6 常量引用

作用:常量引用主要用来修饰形参,防止误操作

在函数形参列表中,可以加const修饰形参,防止形参改变实参

示例:

int main()
{//常量引用//使用场景:用来修饰形参,防止误操作int a = 10;const int& ref = 10;// 引用必须引一块的内存空间// 加上const之后 编译器将代码修改为 // int temp = 10; const int & ref = temp;//ref = 20;//加入const之后,变为只读,不可修改system("pause");return 0;
}
//引用使用的场景,通常用来修饰形参
void showValue(const int& v) {//v += 10;cout << v << endl;
}int main() {const int& ref = 10;//ref = 100;  //加入const后不可以修改变量cout << ref << endl;//函数中利用常量引用防止误操作修改实参int a = 10;showValue(a);system("pause");return 0;
}

1.7引用和指针的区别

对比了引用和指针在C++中的基本性质、初始化要求、空值、操作灵活性、可复制性、安全性和取地址操作等方面的特点:

特性引用 (Reference)指针 (Pointer)
基本性质别名,共享内存地址存储另一个变量地址的变量
初始化要求必须初始化,且不能更改所引用对象可以不初始化,初始化后可更改指向
空值不能指向空值可以指向nullptrNULL
操作灵活性类似普通变量,无算术操作可进行算术操作,改变指向地址
可复制性不可复制,不能重新赋值可复制,可赋值
安全性更高,不易出错,无空指针问题更易出错,如空指针解引用
取地址操作不能直接取引用对象的地址可以取指针本身的地址,可解引用

总的来说,引用和指针在语法和用法上有明显的区别。引用提供了更高级别的抽象和安全性,但牺牲了灵活性;而指针则提供了更低级别的操作和更大的灵活性,但也需要更多的注意来避免潜在的问题。在设计程序时,应根据具体需求和上下文来选择使用引用还是指针。

二、函数提高

2.1 函数默认参数

在C++中,函数的形参列表中的形参是可以有默认值的。

语法:返回值类型 函数名 (参数= 默认值){}

示例:

//函数默认参数
//在C++中,函数的形参列表中的形参是可以有默认值的。
//如果我们自己传入数据,就用自己的数据,如果没有,那么用默认值
//语法: 返回值类型 函数名(形参 = 默认值)
int func(int a, int b = 10, int c = 10) 
{return a + b + c;
} //注意事项
//1. 如果某个位置参数有默认值,那么从这个位置往后,从左向右,必须都要有默认值
//int func2(int a = 10, int b, int c, int d = 10) //error
int func2(int a, int b, int c, int d = 10)
{return a + b + c;
}// 2. 声明和实现只能有一个默认参数
// (如果函数声明有默认值,函数实现的时候就不能有默认参数)
// (如果函数实现有默认值,函数声明的时候就不能有默认参数)
int func2(int a = 10, int b = 10);
int func2(int a, int b)
{return a + b;
}int main() {//cout << func(10) << endl;cout << "ret = " << func(20, 20) << endl;cout << "ret = " << func(100) << endl;system("pause");return 0;
}

2.2函数占位参数

C++中函数的形参列表里可以有占位参数,用来做占位,调用函数时必须填补该位置

语法: 返回值类型 函数名 (数据类型){}

在现阶段函数的占位参数存在意义不大,但是后面的课程中会用到该技术

示例:

//占位参数
//返回值类型 函数名(数据类型){}//函数占位参数 ,占位参数也可以有默认参数
void func(int a, int) {cout << "this is func" << endl;
}int main() {func(10, 10); //占位参数必须填补system("pause");return 0;
}

2.3 函数重载

作用:函数名可以相同,提高复用性

函数重载满足条件:

  • 同一个作用域下

  • 函数名称相同

  • 函数参数类型不同 或者 个数不同 或者 顺序不同

注意: 函数的返回值不可以作为函数重载的条件

示例:

//函数重载
//可以让函数名相同,提高服用型//函数重载的满足条件
//1.同一个作用域下
//2.函数名称相同
//3.函数参数类型不同,或者个数不同,或者顺序不同
void func()
{cout << "func 的调用!" << endl;
}
void func(int a)
{cout << "func (int a) 的调用!" << endl;
}
void func(double a)
{cout << "func (double a)的调用!" << endl;
}
void func(int a, double b)
{cout << "func (int a ,double b) 的调用!" << endl;
}
void func(double a, int b)
{cout << "func (double a ,int b)的调用!" << endl;
}//注意事项
//函数返回值不可以作为函数重载条件(无法重载仅按返回值类型区分的函数)
//int func(double a, int b)
//{
//	cout << "func (double a ,int b)的调用!" << endl;
//}//根据不同的参数运行不同的代码片段
int main() {func();func(10);func(3.14);func(10, 3.14);func(3.14, 10);system("pause");return 0;
}

2.4函数重载注意事项

  • 引用作为重载条件

  • 函数重载碰到函数默认参数

示例:

//函数重载注意事项
//引用作为重载条件
//函数重载碰到函数默认参数
//函数重载注意事项
//1、引用作为重载条件void func(int& a) // int &a = 10; 不合法
{cout << "func (int &a) 调用 " << endl;
}void func(const int& a) //const int &a = 10; 合法
{cout << "func (const int &a) 调用 " << endl;
}//2、函数重载碰到函数默认参数void func2(int a, int b = 10)
{cout << "func2(int a, int b = 10) 调用" << endl;
}void func2(int a)
{cout << "func2(int a) 调用" << endl;
}int main() {int a = 10;func(a); //调用无constfunc(10);//调用有const//func2(10); //当函数重载碰到默认参数,出现二义性,报错,尽量避免这种情况//写函数重载的时候尽量不要写默认参数system("pause");return 0;
}

今天就先到这了!!!

看到这里了还不给博主扣个:
⛳️ 点赞☀️收藏 ⭐️ 关注!

你们的点赞就是博主更新最大的动力!
有问题可以评论或者私信呢秒回哦。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/675521.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【RT-DETR进阶实战】利用RT-DETR进行视频划定区域目标统计计数

👑欢迎大家订阅本专栏,一起学习RT-DETR👑 一、本文介绍 Hello,各位读者,最近会给大家发一些进阶实战的讲解,如何利用RT-DETR现有的一些功能进行一些实战, 让我们不仅会改进RT-DETR,也能够利用RT-DETR去做一些简单的小工作,后面我也会将这些功能利用PyQt或者是…

备战蓝桥杯---搜索(完结篇)

再看一道不完全是搜索的题&#xff1a; 解法1&#xff1a;贪心并查集&#xff1a; 把冲突事件从大到小排&#xff0c;判断是否两个在同一集合&#xff0c;在的话就返回&#xff0c;不在的话就合并。 下面是AC代码&#xff1a; #include<bits/stdc.h> using namespace …

LeetCode-第28题-找出字符串中第一个匹配项的下标

1.题目描述 给你两个字符串 haystack 和 needle &#xff0c;请你在 haystack 字符串中找出 needle 字符串的第一个匹配项的下标&#xff08;下标从 0 开始&#xff09;。如果 needle 不是 haystack 的一部分&#xff0c;则返回 -1 。 2.样例描述 3.思路描述 可以让字符串 …

【前端web入门第四天】01 复合选择器与伪类选择器

文章目录: 1. 复合选择器 1.1 后代选择器 1.2 子代选择器 1.3 并集选择器1.4 交集选择器(了解) 2.伪类选择器 2.1 伪类-文本2.2 伪类-超链接&#xff08;拓展) 1. 复合选择器 什么叫复合选择器? 由两个或多个基础选择器&#xff0c;通过不同的方式组合而成。 复合选择器的作…

158基于matlab的用于分析弧齿锥齿轮啮合轨迹的程序

基于matlab的用于分析弧齿锥齿轮啮合轨迹的程序&#xff0c;输出齿轮啮合轨迹及传递误差。程序已调通&#xff0c;可直接运行。 158 matlab 弧齿锥齿轮啮合轨迹 传递误差 (xiaohongshu.com)

RedissonClient妙用-分布式布隆过滤器

目录 布隆过滤器介绍 布隆过滤器的落地应用场景 高并发处理 多个过滤器平滑切换 分析总结 布隆过滤器介绍 布隆过滤器&#xff08;Bloom Filter&#xff09;是1970年由布隆提出的。它实际上是一个很长的二进制向量和一系列随机映射函数。布隆过滤器可以用于检索一个元素是…

SolidWorks的抽壳

抽壳在建模的时候是比较常见的要求&#xff0c;这里废话不多说&#xff0c;直接开始实例操作。 文章目录 一、规则实体抽壳2、多面抽壳3、空心化抽壳 二、椎体抽壳三、不规则实体抽壳1、不规则实体2、部分实体抽壳 文章随时可能更新&#xff0c;请关注文章原出处&#xff1a; …

JVM之GC垃圾回收

GC垃圾回收 如何判断对象可以回收 引用计数法 如果有对象引用计数加一&#xff0c;没有对象引用&#xff0c;计数减一&#xff0c;如果计数为零&#xff0c;则回收 但是如果存在循环引用&#xff0c;即A对象引用B对象&#xff0c;B对象引用A对象&#xff0c;会造成内存泄漏 可…

Gitlab和Jenkins集成 实现CI (二)

Gitlab和Jenkins集成 实现CI (一) Gitlab和Jenkins集成 实现CI (二) Gitlab和Jenkins集成 实现CI (三) 配置Gitlab api token 配置 Gitlab 进入gitlab #mermaid-svg-t84fR8wrT4sB4raQ {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:…

【芯片设计- RTL 数字逻辑设计入门 6 -- 带同步复位的D触发器 RTL实现及testbench 验证】

文章目录 带同步复位的D触发器Verilog 代码testbench 代码编译及仿真问题小结 带同步复位的D触发器 同步复位 &#xff1a;复位只能发生在在clk信号的上升沿&#xff0c;若clk信号出现问题&#xff0c;则无法进行复位。 Verilog 代码 // timescale ins/1nsmodule flopr (inpu…

Django(十)

1. Ajax请求 浏览器向网站发送请求时&#xff1a;URL 和 表单的形式提交。 GETPOST 特点&#xff1a;页面刷新。 除此之外&#xff0c;也可以基于Ajax向后台发送请求&#xff08;偷偷的发送请求&#xff09;。 依赖jQuery编写ajax代码 $.ajax({url:"发送的地址"…

电脑服务器离线安装.net framework 3.5解决方案(错误:0x8024402c )(如何确定当前系统是否安装NET Framework 3.5)

问题环境&#xff1a; 日常服务的搭建或多或少都会有需要到NET Framework 3.5的微软程序运行框架&#xff0c;本次介绍几种不同的安装方式主要解决运行在Windows 2012 以上的操作系统的服务。 NET Framework 3.5 是什么&#xff1f; .NET Framework是微软公司推出的程序运行框架…

vscode +markdown 的安装和使用

文章目录 前言一、vscode markdown 是什么&#xff1f;1.vscode是什么&#xff1f;2.markdown 是什么&#xff1f; 二、安装步骤1.下载2.安装 三、安装插件1.安装 Markdown All in One2.安装 Markdown Preview Enhanced3. Paste Image v1.0.44.LimfxCodeExv0.7.105.Code Spell …

问题:孔隙比总是1.0。 #知识分享#微信

问题&#xff1a;孔隙比总是1.0。 /ananas/latex/p/1242 参考答案如图所示

深度学习技巧应用36-深度学习模型训练中的超参数调优指南大全,总结相关问题与答案

大家好,我是微学AI,今天给大家介绍一下深度学习技巧应用36-深度学习模型训练中的超参数调优指南大全,总结相关问题与答案。深度学习模型训练中的调优指南大全概括了数据预处理、模型架构设计、超参数优化、正则化策略和训练技巧等多个关键方面,以提升模型性能和泛化能力。 …

【Java】苍穹外卖 Day01

苍穹外卖-day01 课程内容 软件开发整体介绍苍穹外卖项目介绍开发环境搭建导入接口文档Swagger 项目整体效果展示&#xff1a; 管理端-外卖商家使用用户端-点餐用户使用当我们完成该项目的学习&#xff0c;可以培养以下能力&#xff1a; 1. 软件开发整体介绍 作为一名软件开…

在 MacOS 上虚拟化 x86Linux 的最佳方法(通过 Rosetta)

categories: [VM] tags: MacOS VM 写在前面 买了 ARM 的 mac, 就注定了要折腾一下虚拟机了… 之前写过一篇文章是通过 utm 虚拟化archlinux, 其实本质上还是调用了 qemu-system-x86_64, 所以速度并不快, 后来想着能不能借用 Rosetta 的优势即原生转译, 来虚拟化 Intel 的 Linu…

Vue.js2+Cesium1.103.0 十五、绘制视锥,并可实时调整视锥姿态

Vue.js2Cesium1.103.0 十五、绘制视锥&#xff0c;并可实时调整视锥姿态 Demo <template><divid"cesium-container"style"width: 100%; height: 100%;"/> </template><script> /* eslint-disable no-undef */ /* eslint-disable …

MySQL篇----第十五篇

系列文章目录 文章目录 系列文章目录前言一、实践中如何优化 MySQL二、优化数据库的方法三、简单描述 MySQL 中,索引,主键,唯一索引,联合索引的区别,对数据库的性能有什么影响(从读写两方面)前言 前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分…

智慧自助餐饮系统(SpringBoot+MP+Vue+微信小程序+JNI+ncnn+YOLOX-Nano)

一、项目简介 本项目是配合智慧自助餐厅下的一套综合系统&#xff0c;该系统分为安卓端、微信小程序用户端以及后台管理系统。安卓端利用图像识别技术进行识别多种不同菜品&#xff0c;识别成功后安卓端显示该订单菜品以及价格并且生成进入小程序的二维码&#xff0c;用户扫描…