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使用Tenacity一行代码实现自动重试详解

《Python使用Tenacity一行代码实现自动重试详解》tenacity是一个专为Python设计的通用重试库,它的核心理念就是用简单、清晰的方式,为任何可能失败的操作添加重试能力,下面我们就来看... 目录一切始于一个简单的 API 调用Tenacity 入门:一行代码实现优雅重试精细控制:让重试按我

MySQL中EXISTS与IN用法使用与对比分析

《MySQL中EXISTS与IN用法使用与对比分析》在MySQL中,EXISTS和IN都用于子查询中根据另一个查询的结果来过滤主查询的记录,本文将基于工作原理、效率和应用场景进行全面对比... 目录一、基本用法详解1. IN 运算符2. EXISTS 运算符二、EXISTS 与 IN 的选择策略三、性能对比

MySQL常用字符串函数示例和场景介绍

《MySQL常用字符串函数示例和场景介绍》MySQL提供了丰富的字符串函数帮助我们高效地对字符串进行处理、转换和分析,本文我将全面且深入地介绍MySQL常用的字符串函数,并结合具体示例和场景,帮你熟练... 目录一、字符串函数概述1.1 字符串函数的作用1.2 字符串函数分类二、字符串长度与统计函数2.1

使用Python构建智能BAT文件生成器的完美解决方案

《使用Python构建智能BAT文件生成器的完美解决方案》这篇文章主要为大家详细介绍了如何使用wxPython构建一个智能的BAT文件生成器,它不仅能够为Python脚本生成启动脚本,还提供了完整的文... 目录引言运行效果图项目背景与需求分析核心需求技术选型核心功能实现1. 数据库设计2. 界面布局设计3

使用IDEA部署Docker应用指南分享

《使用IDEA部署Docker应用指南分享》本文介绍了使用IDEA部署Docker应用的四步流程:创建Dockerfile、配置IDEADocker连接、设置运行调试环境、构建运行镜像,并强调需准备本... 目录一、创建 dockerfile 配置文件二、配置 IDEA 的 Docker 连接三、配置 Do

Android Paging 分页加载库使用实践

《AndroidPaging分页加载库使用实践》AndroidPaging库是Jetpack组件的一部分,它提供了一套完整的解决方案来处理大型数据集的分页加载,本文将深入探讨Paging库... 目录前言一、Paging 库概述二、Paging 3 核心组件1. PagingSource2. Pager3.

python使用try函数详解

《python使用try函数详解》Pythontry语句用于异常处理,支持捕获特定/多种异常、else/final子句确保资源释放,结合with语句自动清理,可自定义异常及嵌套结构,灵活应对错误场景... 目录try 函数的基本语法捕获特定异常捕获多个异常使用 else 子句使用 finally 子句捕获所

C++11范围for初始化列表auto decltype详解

《C++11范围for初始化列表autodecltype详解》C++11引入auto类型推导、decltype类型推断、统一列表初始化、范围for循环及智能指针,提升代码简洁性、类型安全与资源管理效... 目录C++11新特性1. 自动类型推导auto1.1 基本语法2. decltype3. 列表初始化3

C++11右值引用与Lambda表达式的使用

《C++11右值引用与Lambda表达式的使用》C++11引入右值引用,实现移动语义提升性能,支持资源转移与完美转发;同时引入Lambda表达式,简化匿名函数定义,通过捕获列表和参数列表灵活处理变量... 目录C++11新特性右值引用和移动语义左值 / 右值常见的左值和右值移动语义移动构造函数移动复制运算符

Python对接支付宝支付之使用AliPay实现的详细操作指南

《Python对接支付宝支付之使用AliPay实现的详细操作指南》支付宝没有提供PythonSDK,但是强大的github就有提供python-alipay-sdk,封装里很多复杂操作,使用这个我们就... 目录一、引言二、准备工作2.1 支付宝开放平台入驻与应用创建2.2 密钥生成与配置2.3 安装ali