当前位置: 首页 > news >正文

C++学习笔记-内联函数使用和含义

引言

内联函数是C++为了优化在函数的调用带来的性能开销而设计的,特别是当函数体很小且频繁调用时,内联函数可以让编译器在调用点直接展开函数体,从而避免了函数调用的开销。

一、内联函数的定义与含义

1.1 定义

内联函数是通过在函数声明或定义前加上inline关键字来声明的。这告诉编译器该函数可能是一个好的内联候选,但请注意,这仅仅是一个请求或建议,编译器最终决定是否将其内联。

inline void myFunction() {  // 函数体  
}

1.2 特点

  • 减少函数调用的开销:函数调用涉及保存和恢复调用状态(如寄存器、栈帧等),这在小函数频繁调用时可能成为性能瓶颈。内联函数通过直接在调用点插入函数体来避免这些开销。
  • 代码膨胀:虽然内联可以减少函数调用的开销,但它也可能导致生成的机器代码量显著增加(即代码膨胀),因为每个调用点都会插入相同的函数体。
  • 编译器优化:编译器会根据自己的判断来决定是否内联某个函数,包括函数的复杂度、大小、调用频率等因素。

二、内联函数的使用方式

2.1 基本用法

在函数定义前加inline关键字:这是最直接的方式,但需要注意的是,内联函数通常需要在头文件中定义(除非使用了其他技术如隐式内联或链接时优化),因为编译器需要在每个调用点看到函数体才能决定是否内联。

// 在头文件中  
inline void myInlineFunction() {  // 函数体  
}

在类定义中定义成员函数:在类定义中直接定义的成员函数默认是内联的(如果它们没有使用inline关键字明确声明为非内联)。

class MyClass {  
public:  void myMethod() { // 默认是内联的  // 方法体  }  
};

2.2 注意事项

  • 避免在构造函数和析构函数中使用内联:虽然技术上可以,但构造函数和析构函数中可能包含复杂的初始化或清理代码,这些代码不适合内联。
  • 避免在大型函数中使用内联:大型函数不适合内联,因为它们会显著增加代码膨胀,而且编译器可能出于各种原因拒绝内联它们。
    递归函数不能内联:因为内联函数需要在调用点直接插入函数体,而递归函数会调用自身,这会导致无限展开。
  • 虚函数和静态成员函数:虚函数不能内联,因为虚函数的调用是通过虚函数表实现的,这涉及到动态绑定。静态成员函数虽然可以内联,但它们并不依赖于类的实例,因此与内联函数的优化目标不完全一致。

三、典型程序示例

以一个简单典型的C++代码示例,展示内联函数的含义和使用方式。

#include <iostream>  // 定义一个内联函数来计算两个整数的和  
inline int add(int x, int y) {  return x + y;  
}  int main() {  int a = 5, b = 3;  // 调用内联函数  int sum = add(a, b);  // 输出结果  std::cout << "The sum of " << a << " and " << b << " is " << sum << std::endl;  return 0;  
}

这个示例中,通过在函数定义前加上inline关键字,我们定义了一个内联函数(add函数)来计算两个整数的和,并在main函数中调用它。这意味着在编译时,编译器会尝试在add函数的每个调用点处直接插入函数体(即return x + y;),从而避免了函数调用的开销。这也就是为什么说内联函数适用“函数体很小且频繁调用”的场景。

注意:

  • inline关键字对编译器来说只是一个请求或建议,编译器最终会根据多种因素(如函数的大小、复杂度、调用频率以及编译器的优化策略)来决定是否真正内联该函数。
    编译器在决定是否内联函数时会考虑多种因素,包括但不限于:
  • 函数的大小:小的函数更有可能被内联。
  • 函数的调用频率:频繁调用的函数更有可能被内联。
  • 函数的复杂度:包含复杂控制流或大量计算的函数可能不适合内联。
  • 编译器的优化级别:通常,在较高的优化级别下,编译器会更积极地尝试内联函数。
  • 此外,虽然在这个例子中我们将add函数的定义放在了头文件中(或者至少是在main函数之前可见的地方),但在实际应用中,如果内联函数需要在多个源文件中使用,通常的做法是将函数声明放在头文件中,并在头文件中使用inline关键字,而在一个源文件中提供函数的定义(不使用inline关键字)。不过,对于小型的、简单的函数来说,直接在头文件中提供定义是一种常见且简单的方法。
http://www.lryc.cn/news/407296.html

相关文章:

  • 数据库(MySQL)-视图、存储过程、触发器
  • js 优雅的实现模板方法设计模式
  • C语言——输入输出
  • 【微软蓝屏】微软Windows蓝屏问题汇总与应对解决策略
  • OpenCV图像滤波(2)均值平滑处理函数blur()的使用
  • Android lmkd机制详解
  • linux shell(中)
  • VMware三种网络模式---巨细
  • 力扣高频SQL 50 题(基础版)第一题
  • 2.1.卷积层
  • 网易《永劫无间》手游上线,掀起游戏界狂潮
  • RNN(一)——循环神经网络的实现
  • php 根据位置的经纬度计算距离
  • 17 Python常用内置函数——基本输入输出
  • 【Web】LitCTF 2024 题解(全)
  • 家政项目小程序的设计
  • electron TodoList网页应用打包成linux deb、AppImage应用
  • 【C语言】 使用fgets和fputs完成两个文件的拷贝
  • 使用PyTorch导出JIT模型:C++ API与libtorch实战
  • Python——异常捕获,传递及其抛出操作
  • 【Maven】 的继承机制
  • 微信小程序结合后端php发送模版消息
  • sqlalchemy报错sqlalchemy.orm.exc.DetachedInstanceError
  • 华为网络模拟器eNSP安装部署教程
  • 【React】详解样式控制:从基础到进阶应用的全面指南
  • 【ROS2】高级:安全-理解安全密钥库
  • C语言 ——— 数组指针的定义 数组指针的使用
  • opencascade AIS_ManipulatorOwner AIS_MediaPlayer源码学习
  • 如何防止用户通过打印功能复制页面文字
  • Python3网络爬虫开发实战(3)网页数据的解析提取