C++内联函数:那时我还太年轻,并不知道使用inline带来的效率,早已在暗中标好了价格

本文主要是介绍C++内联函数:那时我还太年轻,并不知道使用inline带来的效率,早已在暗中标好了价格,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在这里插入图片描述

  • 👑专栏内容:C++学习笔记
  • ⛪个人主页:子夜的星的主页
  • 💕座右铭:日拱一卒,功不唐捐

文章目录

  • 一、前言
  • 二、内联函数
    • 1、起源
    • 2、概念
  • 三、与宏的区别
    • 1、宏的缺点
    • 2、两者区别
  • 四、内联函数的代价
    • 代价一:可执行程序变大
    • 代价二:`inline`可能被忽略
    • 代价三:声明和定义不可分离
  • 五、总结


一、前言

关键字inline是C++相对于C语言的又一个扩充,在函数的声明或定义、函数的返回类型前加上关键字inline,即可把函数指定为内联函数从而提升程序运行的效率。但使用inline是要付出代价的,正如茨威格在《断头王后》中那样写道:“ 那时候她还太年轻,不知道所有命运馈赠的礼物,早已在暗中标好了价格。” 那么inline的优势和它为此要付出的代价是什么呢?让我们来慢慢揭晓!

二、内联函数

1、起源

当一个函数被调用执行时,首先要在栈中为形参和局部变量分配存储空间,然后还要将实参的值复制给形参,接下来还要将函数的返回地址放入栈中,最后才跳转到函数内部执行。这个过程是要消耗时间和栈空间(放置函数内数据的内存空间)的。当一个函数非常短小,但由于被放入循环体中大量的循环,就会消耗大量的时间。同时由于栈空间是有限,所以频繁大量的使用也会造成因栈空间不足所造成的出错。
在这里插入图片描述

2、概念

内联函数是一种编程语言结构,用来建议编译器对一些特殊函数进行内联扩展;也就是说建议编译器将指定的函数体插入并取代每一处调用该函数的地方(上下文),从而节省了每次调用函数带来的额外时间开支。在C++中在一个函数的前面加上inline进行修饰,就会将这个函数变为内联函数。
在这里插入图片描述
小C挑衅大哥也不是一天两天了,本着 能打就打 以理服人的做法。接下来我们就举个例子现场回应他的挑衅。

int Add(int left,int right)
{return left + right;
}
int main ()
{int ret = 0;ret = Add(1,2);return 0;
}

接着,我们点住ret查看其汇编代码。
在这里插入图片描述
我们可以看到,ret的汇编代码中,有一句的前面有一个call,在汇编语言中call就是函数调用指令。它的作用就是将程序当前执行的位置IP压入堆栈中,转移到调用的子程序。
接下来。我们在刚刚的函数前面加入inline进行修饰,再看其汇编语言。

inline int Add(int left,int right)
{return left + right;
}
int main ()
{int ret = 0;ret = Add(1,2);return 0;
}

在这里插入图片描述
哇,汇编代码中的call果然消失不见了呢~
那接下来该干嘛了?当然是以理服人!

在这里插入图片描述

三、与宏的区别

C语言中确实也有不建立栈帧的方法,那就是使用宏来定义函数。

在这里插入图片描述

#define Add(int x,int y) return x+y;

呀…写错了,我重写!

#define Add(x,y) x+y;

额…又错了!再给一次机会!

#define Add(x,y) (x)+(y)

呜…失误!再再给我一次机会!

#define Add(x,y)((x)+(y))

哇…终于写对了!!!!
在这里插入图片描述

1、宏的缺点

宏其实是一种非常暴力的替换,正因为如此,它的缺点有很多:

  1. 宏函数不能进行调试。
  2. 宏函数不会进行类型的检测,代码安全性低。
  3. 宏函数容易产生二义性,可维护性差。
  4. 宏函数的编写容易出现错误。

2、两者区别

  1. 内联函数是在编译时展开,而宏在预编译时展开。
  2. 在编译的时候,内联函数直接被嵌入到目标代码中去,而宏只是一个简单的文本替换。
  3. 内联函数可以进行诸如类型安全检查、语句是否正确等编译功能,宏不具有这样的功能。

四、内联函数的代价

在这里插入图片描述

代价一:可执行程序变大

inline是一种以空间换时间的做法,如果编译器将函数当成内联函数处理,在编译阶段,会
用函数体替换函数调用,这样就会导致编译出来的可执行程序变大。

代价二:inline可能被忽略

inline对于编译器而言只是一个建议,不同编译器关于inline实现机制可能不同。
一般建议:将函数规模较小(即函数不是很长,具体没有准确的说法,取决于编译器内部实现)、不是递归、且频繁调用的函数采用inline修饰,否则编译器会忽略inline特性。
《C++prime》第五版关于inline的建议:
在这里插入图片描述

代价三:声明和定义不可分离

定义和声明分离的后果是什么?以下面的代码为例:

F.h
#include <iostream>
using namespace std;
inline void f(int i);
F.cpp
#include "F.h"
void f(int i)
{cout << i << endl;
}
main.cpp
#include "F.h"
int main()
{f(10);return 0;
}

运行结果:
在这里插入图片描述
为什么普通函数可以将定义和声明分离,而内联函数不行呢?
在这里插入图片描述
看汇编代码,普通函数会产生跳转时对应的地址。而内联函数默认在用的地方已经展开了,不需要产生。所以只有声明的话,在声明的地方已经展开了,这会导致在调用的时候没法展开。

五、总结

优点:

  • 函数使用时进行替换,效率高

缺点:

  • 如果函数的代码较长,使用内联将消耗过多内存
  • 如果函数体内有循环,那么执行函数代码时间比调用开销大

建议:

  • 建议 inline 函数的定义放在头文件中
  • 经常使用且代码短的函数,才进行内联
  • 函数体内的代码比较长时,不使用内联
  • 函数体内出现多重循环时,不使用内联

在这里插入图片描述

这篇关于C++内联函数:那时我还太年轻,并不知道使用inline带来的效率,早已在暗中标好了价格的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/672905

相关文章

Python中模块graphviz使用入门

《Python中模块graphviz使用入门》graphviz是一个用于创建和操作图形的Python库,本文主要介绍了Python中模块graphviz使用入门,具有一定的参考价值,感兴趣的可以了解一... 目录1.安装2. 基本用法2.1 输出图像格式2.2 图像style设置2.3 属性2.4 子图和聚

windows和Linux使用命令行计算文件的MD5值

《windows和Linux使用命令行计算文件的MD5值》在Windows和Linux系统中,您可以使用命令行(终端或命令提示符)来计算文件的MD5值,文章介绍了在Windows和Linux/macO... 目录在Windows上:在linux或MACOS上:总结在Windows上:可以使用certuti

CentOS和Ubuntu系统使用shell脚本创建用户和设置密码

《CentOS和Ubuntu系统使用shell脚本创建用户和设置密码》在Linux系统中,你可以使用useradd命令来创建新用户,使用echo和chpasswd命令来设置密码,本文写了一个shell... 在linux系统中,你可以使用useradd命令来创建新用户,使用echo和chpasswd命令来设

Python使用Matplotlib绘制3D曲面图详解

《Python使用Matplotlib绘制3D曲面图详解》:本文主要介绍Python使用Matplotlib绘制3D曲面图,在Python中,使用Matplotlib库绘制3D曲面图可以通过mpl... 目录准备工作绘制简单的 3D 曲面图绘制 3D 曲面图添加线框和透明度控制图形视角Matplotlib

Pandas中统计汇总可视化函数plot()的使用

《Pandas中统计汇总可视化函数plot()的使用》Pandas提供了许多强大的数据处理和分析功能,其中plot()函数就是其可视化功能的一个重要组成部分,本文主要介绍了Pandas中统计汇总可视化... 目录一、plot()函数简介二、plot()函数的基本用法三、plot()函数的参数详解四、使用pl

使用Python实现IP地址和端口状态检测与监控

《使用Python实现IP地址和端口状态检测与监控》在网络运维和服务器管理中,IP地址和端口的可用性监控是保障业务连续性的基础需求,本文将带你用Python从零打造一个高可用IP监控系统,感兴趣的小伙... 目录概述:为什么需要IP监控系统使用步骤说明1. 环境准备2. 系统部署3. 核心功能配置系统效果展

使用Java将各种数据写入Excel表格的操作示例

《使用Java将各种数据写入Excel表格的操作示例》在数据处理与管理领域,Excel凭借其强大的功能和广泛的应用,成为了数据存储与展示的重要工具,在Java开发过程中,常常需要将不同类型的数据,本文... 目录前言安装免费Java库1. 写入文本、或数值到 Excel单元格2. 写入数组到 Excel表格

redis中使用lua脚本的原理与基本使用详解

《redis中使用lua脚本的原理与基本使用详解》在Redis中使用Lua脚本可以实现原子性操作、减少网络开销以及提高执行效率,下面小编就来和大家详细介绍一下在redis中使用lua脚本的原理... 目录Redis 执行 Lua 脚本的原理基本使用方法使用EVAL命令执行 Lua 脚本使用EVALSHA命令

C#如何调用C++库

《C#如何调用C++库》:本文主要介绍C#如何调用C++库方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录方法一:使用P/Invoke1. 导出C++函数2. 定义P/Invoke签名3. 调用C++函数方法二:使用C++/CLI作为桥接1. 创建C++/CL

Java 中的 @SneakyThrows 注解使用方法(简化异常处理的利与弊)

《Java中的@SneakyThrows注解使用方法(简化异常处理的利与弊)》为了简化异常处理,Lombok提供了一个强大的注解@SneakyThrows,本文将详细介绍@SneakyThro... 目录1. @SneakyThrows 简介 1.1 什么是 Lombok?2. @SneakyThrows