【阅读笔记】c++ Primer Plus——第八章

news2025/1/10 20:47:56

函数探幽

c++内联函数

  • 为了提高程序运行速度而做的改进
  • 编译的最终产物是可执行程序——由一组机器语言指令组成。运行程序时,操作系统将这些指令载入到计算机内存中,因此每条指令都有特定的内存地址。然后计算机开始逐步执行指令。执行到函数调用的时候,程序将在调用函数后立即存储该指令的内存地址,并将函数参数赋值到堆栈,跳到标记函数起点的内存单元,执行函数代码,然后再跳回到地址被保存的指令处。在此过程中对计算器是有一定开销的
  • 内联函数,使用相应的函数替换函数调用,让程序无需调到另一个位置处执行代码,再跳回来。运行速度会比普通函数调用要快,但缺点就是需要更多的内存
  • 使用内联函数
    • 在函数声明前加上关键字inline
    • 在函数定义前加上关键字inline

引用变量

  • 引用是已定义的变量的别名,主要用作函数的形参,通过将引用变量用作参数,函数将使用函数数据,而不是其副本
    创建引用变量

  • &在c++里面不止是取址符,还能用来声明引用

int rat = 10;
int & test = rat;
cout << "rat的值是" << rat << endl;
cout << "test的值是" << test << endl;
rat++;
cout << "test的值是" << test << endl;
cout << &rat << endl;
cout << &test << endl;
//rat的值是10
//test的值是10
//test的值是11
//0x7ffc9b3dfe74
//0x7ffc9b3dfe74

来个图

  • 需要在声明引用的时候进行初始化
int rat = 10;

int & test;
test = rat;
// 会报错
  • 将引用用作函数参数,是函数中的变量名成为调用程序中的变量别名,此传递参数方法称为按引用传递
  • 按引用传递允许被调用的函数能够访问调用函数中的变量
  • 按引用传递和按指针传递函数参数的,都能够对原始数据进行修改。swapr中的变量a,b可以看作是wallet1和wallet2的别名,而swapv则是创建了新的变量a和b,其值和wallet1和wallet2相同
void swapr(int & a, int & b);
void swapp(int *p, int *q);
void swapv(int a, int b);
int main()
{
    using namespace std;
    int wallet1 = 300;
    int wallet2 = 250;

    // const int wallet1 = 300;
    // const int wallet2 = 250;

    cout << wallet1 << "  " << wallet2 << endl;
    swapr(wallet1, wallet2);
    cout << wallet1 << "  " << wallet2 << endl;
    swapp(&wallet1, &wallet2);
    cout << wallet1 << "  " << wallet2 << endl;
    swapv(wallet1, wallet2);    //没有换过来
    cout << wallet1 << "  " << wallet2 << endl;

    return 0;
}
void swapr(int &a, int &b)
{
    int temp;
    temp = a;
    a = b;
    b = temp;
}
void swapp(int *p, int *q)
{
    int temp;
    temp = *p;
    *p = *q;
    *q = temp;
}
void swapv(int a, int b)
{
    int temp;
    temp = a;
    a = b;
    b = a;
}

来个图

  • 在函数引用参数中, 当参数为const引用时,c++会生成临时变量
  • 实参类型不正确,但可以转换为正确类型
  • 实参类型正确,但不是左值(左值参数是可被引用的数据对象,如变量、数组元素、结构成员、引用和解除引用的指针)
  • 引用参数的声明尽可能地去使用const

    • 可以避免无意中修改数据的编程错误
    • 使用const是函数能够处理const和非const实参,否则只能接受非const数据
    • 使用const引用使函数能够正确生成并使用临时变量
double refcube(const double &ra);
int main()
{
    double side = 3.0;
    double * pd = &side;
    double &rb = side;
    long edge = 5L;
    double lens[4] = {1.0, 3.0, 5.0, 6.0};
double c1 = refcube(side);
double c2 = refcube(*pd);
double c3 = refcube(rb);
double c5 = refcube(edge);
double c4 = refcube(lens[2]);
double c6 = refcube(7.0);
double c7 = refcube(side + 10.0);
cout &lt;&lt; c1 &lt;&lt; endl;
cout &lt;&lt; c2 &lt;&lt; endl;
cout &lt;&lt; c3 &lt;&lt; endl;
cout &lt;&lt; c4 &lt;&lt; endl;
cout &lt;&lt; c5 &lt;&lt; endl;
cout &lt;&lt; c6 &lt;&lt; endl;
cout &lt;&lt; c7 &lt;&lt; endl;
return 0;
}
double refcube(const double &ra)
{
    return ra*ra*ra;
}
  • 使用引用参数的主要原因是
  • 程序有可能会修改调用函数中的数据对象

  • 通过传递引用而不是整个数据对象,可提高程序的运行速度
    使用引用参数的情况:

    • 数据对象小(内置数据类型),可使用按值传递
    • 数据对象是数组,使用指针
    • 数组对象是较大的结构,使用const指针或const引用
    • 数组对象是类对象,使用const引用
  • 对于修改调用函数中的数据的函数
    • 数据对象是内置数据类型,则使用指针
    • 数据对象是数组,则使用指针
    • 数据对象是结构,则使用引用或指针
    • 数据对象是类对象,则使用引用
  • 将引用用于结构,使用结构引用参数的方式与使用基本变量引用相同,只需在声明结构参数时使用引用运算符&即可
struct free_thorws
{
    int made;
    int attempts;
    float percent;
}
void get_test(free_thorws & ft);
  • 像下面free_throw &则该函数返回的是引用,free_throw的话,则是直接返回一个值
free_throw & accumulate(free_throws & target, const free_throws & source);
  • 返回引用与返回值不同的区别
    • 返回值将return的内容,赋值到一个临时的位置,然后程序调用的时候用的就是这个值
    • 返回引用是直接使用该内存地址上面的值,相对来说效率会提高一点
  • 返回引用是需要注意的是,要避免返回函数终止时不再存在的内存单元引用。下面的newguy在函数运行完毕之后就不再存在了,解决方法是可以用new来分配新的内存空间给newguy并返回指向该内存空间的指针
const free_thorw & clone2(free_throws & ft)
{
    free_throws newguy;
    newguy = ft;
    return newguy;
}

默认参数

  • 对参数列表的函数,必须从左到右添加默认值
int harpo(int n, int m = 6, int j = 7);
int chico(int n, int m = 6, int j);

在这里插入图片描述

函数重载

  • 函数重载能够让程序使用多个同名的函数。若没有匹配的原型并不会自动通知使用其中的某个函数,c++会将其类型强制进行转换再匹配,要是转换后有多重匹配结构,则拒绝调用并报错
  • 仅当函数基本上执行相同的任务,但使用不同形式的数据时,才应采用重载
void print(const char * str, int width);
void print(double d, int width);    //这里注释掉其中两个函数则能正常运行
void print(long l, int width);
void print(int i, int width);
void print(const char *str);
int main()
{
    print("asdsad", 15);
    print("Sdjosia");
    print(199.0, 10);
    print(1992, 12);
    print(1993L, 15);
    unsigned int year = 6;
    print(year, 22);    //这里会报错
    return 0;
}
void print(const char * str, int width)
{
    cout << str << " "<<width << endl;
}
void print(double d, int width)
{
    cout << d << " " << width << endl;
}
void print(long l, int width)
{
    cout << l << " " << width << endl;
}
void print(int i, int width)
{
    cout << i << " " << width << endl;
}
void print(const char *str)
{
    cout << str << endl;
}
  • c++跟踪每个重载函数的原理是编译器根据函数原型中指定的形参类型对每个函数名进行加密

函数模板

  • 使用泛型来定义函数,通过泛型来对具体类型进行替换
template <typename T> void Swap(T &a, T&b);
int main()
{
    int i = 10;
    int j = 20;
    Swap(i, j);
    cout << i << " " << j << endl;
    float m = 10.3;
    float n = 23.5;
    Swap(m,n);
    cout << m << " " << n << endl;
    return 0;
}

template <typename T> void Swap(T &a, T &b)
{
    T temp;
    temp = a;
    a = b;
    b = temp;
}
  • 模板重载,与重载常规函数定义一样
template <typename T> void Swap(T &a, T&b);
template <typename T> void Swap(T *a, T *b, int n);
void show(int a[]);
const int Lim = 8;
int main()
{
    int i = 10, j = 20;
    Swap(i, j);
    cout << i << " " << j<< endl;

    int d1[Lim] = {0, 7, 0, 4, 1, 7, 7, 6};
    int d2[Lim] = {0, 7, 2, 0, 1, 9, 6, 9};
    show(d1);
    show(d2);
    Swap(d1, d2, Lim);
    show(d1);
    show(d2);
    return 0;
}

template <typename T> void Swap(T &a, T &b)
{
    T temp;
    temp = a;
    a = b;
    b = temp;
}
template <typename T> void Swap(T a[], T b[], int n)
{
    T temp;
    for (int i = 0; i< n; i++)
    {
        temp = a[i];
        a[i] = b[i];
        b[i] = temp;
    }
}
void show(int a[])
{
    cout << a[0] << a[1] << "/";
    cout << a[2] << a[3] << "/";
    for (int i = 4; i<Lim; i++)
    {
        cout << a[i];
    }
    cout << endl;
}

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

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

相关文章

一道编程劝退题,检测你是否适合干编程

前言大家都知道要想成为一名优秀的开发工程师&#xff0c;需要数学基础好&#xff0c;即你要有很强的逻辑思维能力&#xff0c;这里有一道美国斯坦福大学出的一道逻辑思维的测试测试&#xff0c;检测你的逻辑思维能力&#xff0c;大家可以看看自己逻辑能力怎么样。题目有一个抽…

<队列>的概念结构实现【C语言版】

1.队列的概念及结构 队列对于临时数据的处理也十分有趣&#xff0c;它跟栈一样都是有约束条件的数组&#xff08;或者链表&#xff09;。区别在于我们想要按什么顺序去处理数据&#xff0c;而这个顺序当然是要取决于具体的应用场景。 你可以将队列想象成是电影院排队。排在最…

Android Studio 工程导入 AOSP编译的 android.jar

使用场景   1.需要使用 framework 中的 SystemApi 文件或者 hide 的 API 接口   2.定制 Framework 层业务&#xff0c;即有 客制化 的 API 接口 补充知识点   1.framework 源码即 AS 工程目录中 External Libraries 下的 < Android API xx Platform > 下的原生 SDK…

Docker-全面详解(学习总结---从入门到深化)

一、什么是Docker Docker 是一个开源的应用容器引擎&#xff0c;基于 Go 语言 并遵从 Apache2.0 协议开源。 什么是"集装箱技术" 我们都知道码头里的集装箱是运载货物用的&#xff0c;它是一种按规格标准 化的钢制箱子。集装箱的特色&#xff0c;在于其格式划一&…

峰值21WQps、亿级DAU,小游戏《羊了个羊》是怎么架构的?

小游戏《羊了个羊》 短短的7天内&#xff0c;DAU突破了1亿、吞吐量峰值21WQps。 《羊了个羊》运营后台数据显示&#xff0c;在短短的7天内&#xff0c;这款小游戏的DAU就突破了1亿。 要知道&#xff0c;除了王者荣耀、原神等屈指可数的现象级手游之外&#xff0c;1亿DAU是这个…

LeetCode刷题模版:131 - 140

目录 简介131. 分割回文串132. 分割回文串 II133. 克隆图134. 加油站135. 分发糖果136. 只出现一次的数字137. 只出现一次的数字 II138. 复制带随机指针的链表139. 单词拆分140. 单词拆分 II【未理解】结语简介 Hello! 非常感谢您阅读海轰的文章,倘若文中有错误的地方,欢迎您…

【GitHub仓库上传和克隆】

GitHub仓库上传和克隆1. 安装1.1 WindowR:cmd1.2 选择写代码的文件夹右键Git Bash Here1.3 分支管理2. 本地仓库推送给网络仓库2.1 忽略推送文件2.2 本地初始推送3. 克隆3.1 克隆下载3.2 克隆代码修改上传3.2.1 初次推送3.2. 2 后续推送其他1. 安装 1.1 WindowR:cmd //是否安…

Redis从入门到精通(一:Redis入门)

Redis 简介 问题现象 海量用户 高并发 罪魁祸首——关系型数据库性能瓶颈&#xff1a;磁盘IO性能低下扩展瓶颈&#xff1a;数据关系复杂&#xff0c;扩展性差&#xff0c;不便于大规模集群解决思路降低磁盘IO次数&#xff0c;越低越好 —— 内存存储去除数据间关系&#xff0…

ucore lab1,lab2,lab3,lab4链表详解 获取结构体成员偏移

ucore版链表介绍 ucore是清华大学操作系统实验课要完成的操作系统&#xff0c;里面有个链表数据结构我觉得很有意思&#xff0c;记录下来。 ucore将链表与数据对象分离&#xff0c;使得任意数据对象&#xff0c;只要加上一个链表组件就能组织成一个链表。 要使得一个本来不具…

双目相机国产、非国产统计参数对比分析

双目相机国产、非国产统计参数对比分析 ZED ZED是STEREOLABS出品的双目摄像头&#xff0c;广泛应用在科研机构的无人车、协作机械臂上&#xff0c;如图2-1所示。其3D分辨率在Ultra模式下可达到RGB时的分辨率&#xff0c;具体见图2-2&#xff0c;物理尺寸为1753033mm&#xff…

记录一次gateway HandlerStrategies.withDefaults().messageReaders() 导致的内存炸裂的问题

背景 年前出现了一次内存炸裂的生产事故。导致其他请求无法请求通过。 [boundedElastic-55] [Loggers.java:314]:Scheduler worker in group main failed with an uncaught exception [TID:] 2023-01-18 10:40:33.189 [INFO] [boundedElastic-55] [AccessTokenGatewayFilterF…

Arduino基础学习——meArm(太极创客第二部分)

面包板电源模块为机器臂单独供电&#xff0c;机器臂本身有四个小电机驱动作用&#xff0c;如果单独靠arduino来为这四个小电机供电&#xff0c;机器臂可能不会稳定工作&#xff0c;将会抖动。 机械臂的四个动作主要靠四个电机来控制&#xff0c;这四个电机主要连接在我们的ard…

云端IDE系列教程4:TitanIDE + Typora = 鱼和熊掌

概述 目前&#xff0c;大部分技术人员使用 Markdown 编写技术文档已经成了日常工作的一部分&#xff0c;现在市场上也有各种各样的文字编辑工具&#xff1a;石墨文档、有道云笔记、语雀、金山文档、腾讯文档、Google文档&#xff0c;WPS、Office、Typora等。但在云原生时代&am…

vue3学习笔记(总)——ts+组合式API(setup语法糖)

文章目录1. ref全家桶1.1 ref()1.2 isRef()以及isProxy()1.3 shallowRef()1.4 triggerRef()1.5 customRef()1.6 unref()2. reactive全家桶2.1 reactive()2.2 readonly()2.3 shallowReactive() 和 shallowReadonly()3. to系列全家桶3.1 toRef()3.2 toRefs()3.3 toRaw()4. comput…

【年度总结】回看2022,展望2023,做更好的自己

作者主页&#xff1a;Designer 小郑 作者简介&#xff1a;Java全栈软件工程师一枚&#xff0c;来自浙江宁波&#xff0c;负责开发管理公司OA项目&#xff0c;专注软件前后端开发&#xff08;Vue、SpringBoot和微信小程序&#xff09;、系统定制、远程技术指导。CSDN学院、蓝桥云…

Mac 下ZooKeeper安装和使

Mac 下ZooKeeper安装和使用 Apache ZooKeeper分布式协调系统是构建分布式应用程序的高性能服务。 1.下载ZooKeeper 环境要求&#xff1a;ZooKeeper服务器是用Java创建的&#xff0c;它运行在JVM之上。需要安装JDK 7或更高版本。 https://zookeeper.apache.org/releases.html …

1.3 PCIe——硬件实现架构

PCIe的设计可以分为controller和PHY&#xff0c;整体设计较为复杂&#xff0c;一般可向IP厂商定制设计&#xff0c;controller和PHY模块的接口是PIPE接口 一、一般实现架构 1.1 PCIE controller 控制器逻辑包含了IP的host设计&#xff0c;以及PCIe协议中所规定的事务层、数…

AirServer电脑投屏软件免费版使用及切换中文教程

AirServer是由App Dynamic打造的一款投屏软件。AirServer是适用于Mac和Windows的先进的屏幕镜像接收器。可以将手机设备&#xff0c;如iPhone、iPad、安卓上的屏幕投送到电脑屏幕上。特别我们日常开会要给客户演示手机上的操作时&#xff0c;投屏就显得非常专业。当然&#xff…

关于java位移运算的一点讨论

框架乱飞的年代&#xff0c;时常还得往框架源码里看&#xff0c;对内在原理没点理解&#xff0c;人家就会认为你不太行。平时开发你可能没咋用过位移运算&#xff0c;但往源码里一看&#xff0c;就时常能看到它。我也是看着看着&#xff0c;突然仔细一琢磨&#xff0c;又不由得…

【机器学习 - 4】:线性回归算法

文章目录线性回归线性回归的理解损失函数简单线性回归封装线性回归算法线性回归算法在sklearn中调用线性回归算法向量化运算线性回归模型中的误差均方误差 MSE均方根误差平均绝对误差调用sklearn中的均方根误差和平均绝对误差函数R squared error &#xff08;常用&#xff09;…