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

C语言基础系列【19】inline关键字

博主介绍:程序喵大人

  • 35- 资深C/C++/Rust/Android/iOS客户端开发
  • 10年大厂工作经验
  • 嵌入式/人工智能/自动驾驶/音视频/游戏开发入门级选手
  • 《C++20高级编程》《C++23高级编程》等多本书籍著译者
  • 更多原创精品文章,首发gzh,见文末
  • 👇👇记得订阅专栏,以防走丢👇👇
    😃C++基础系列专栏
    😃C语言基础系列
    😃C++大佬养成攻略

关于inline,我们直接了解以下几个知识点即可。

inline是一个请求(而非命令)

inline关键字用于向编译器发出一个请求,建议将函数体在每个调用点内联展开。

这意味着编译器在编译过程中,可能会将函数的代码直接插入到调用该函数的地方,而不是通过通常的函数调用机制来执行。

需要注意的是,inline只是一个建议,编译器可以选择是否接受这个建议。

编译器可能会基于多种因素(如函数的大小、复杂性、调用频率以及整体代码的优化目标)来决定是否进行内联展开。

inline函数通常用于小函数

inline函数通常用于那些执行速度快且调用频繁的小函数。这些函数通常只有几行代码,并且不包含复杂的控制结构或大量的计算。

通过将这些小函数内联展开,可以减少函数调用的开销(如栈帧的创建和销毁、参数传递等),从而提高程序的执行效率。

对于较大的函数或包含复杂逻辑的函数,内联展开可能会导致代码膨胀,甚至可能降低性能。

inline函数的定义通常放在头文件中

由于inline函数需要在每个调用点展开,因此其定义需要在编译时对每个编译单元可见。这通常意味着inline函数的定义应该放在头文件中,而不是源文件(.c)中。这样做可以确保在链接时不会出现重复定义的问题,因为每个编译单元都会包含inline函数的定义,并且编译器会处理这些重复的定义(实际上,由于inline的特性,编译器会将其视为建议而非强制要求,因此不会有链接时的符号冲突)。

如果inline函数在多个编译单元中被引用,并且没有通过static关键字限制其链接性,那么在某些编译器和链接器实现中可能会遇到链接错误。为了避免这种情况,通常建议将inline函数声明为static,从而限制其在单个编译单元内的可见性。

inline函数不能包含复杂的控制结构

inline函数通常应该避免包含复杂的控制结构,如循环和递归。这是因为内联展开这些结构可能会导致代码膨胀,增加程序的内存占用,并且可能不会带来性能上的提升。

特别是递归函数,由于递归调用本身的开销和栈空间的使用,内联展开通常是不合适的。

对于包含复杂控制结构的函数,即使它们很小,编译器也可能选择不进行内联展开。

编译器可能忽略inline请求

编译器可能会忽略inline请求,特别是在以下情况下:

  • 函数体较大:如果函数体包含大量代码,编译器可能会认为内联展开会导致代码膨胀,从而选择不进行内联。
  • 包含复杂逻辑:如果函数包含复杂的控制结构、大量的计算或条件分支,编译器可能会认为内联展开不会带来性能上的优势,甚至可能降低性能。
  • 优化级别:编译器的优化级别也会影响其是否接受inline请求。在较低的优化级别下,编译器可能会更加保守地选择不进行内联展开。
  • 链接时优化:在某些情况下,即使函数在源代码中被标记为inline,编译器也可能在链接时决定不进行内联展开。这取决于编译器的链接时优化能力和策略。

验证是否inline

看这段代码:

#include <stdio.h>
#include <time.h>
#include <stdint.h>

// 定义一个inline函数
inline int add_inline(int a, int b) {
    return a + b;
}

// 定义一个普通函数
int add_normal(int a, int b) {
    return a + b;
}

int main() {
    const double N = 1000000000000000000; // 函数调用次数
    clock_t start, end;
    double cpu_time_used;

    // 测试普通函数
    start = clock();
    for(double i = 0; i < N; i += 0.0000000001) {
        add_normal(i, i);
    }
    end = clock();
    cpu_time_used = ((double) (end - start)) / CLOCKS_PER_SEC;
    printf("Normal function took %f seconds to execute \n", cpu_time_used);

    // 测试inline函数
    start = clock();
    for(double i = 0; i < N; i += 0.0000000001) {
        add_inline(i, i);
    }
    end = clock();
    cpu_time_used = ((double) (end - start)) / CLOCKS_PER_SEC;
    printf("Inline function took %f seconds to execute \n", cpu_time_used);

    

    return 0;
}

汇编代码:

add_normal(int, int):
 lea    eax,[rdi+rsi*1]
 ret
 data16 data16 cs nop WORD PTR [rax+rax*1+0x0]
main:
 push   rbx
 call   1040 <clock@plt>
 mov    rbx,rax
 call   1040 <clock@plt>
 sub    rax,rbx
 cvtsi2sd xmm0,rax
 divsd  xmm0,QWORD PTR [rip+0xe8a]        # 2008 <_IO_stdin_used+0x8>
 lea    rdi,[rip+0xe8b]        # 2010 <_IO_stdin_used+0x10>
 mov    al,0x1
 call   1030 <printf@plt>
 call   1040 <clock@plt>
 mov    rbx,rax
 call   1040 <clock@plt>
 sub    rax,rbx
 xorps  xmm0,xmm0
 cvtsi2sd xmm0,rax
 divsd  xmm0,QWORD PTR [rip+0xe5c]        # 2008 <_IO_stdin_used+0x8>
 lea    rdi,[rip+0xe8a]        # 203d <_IO_stdin_used+0x3d>
 mov    al,0x1
 call   1030 <printf@plt>
 xor    eax,eax
 pop    rbx
 ret

代码中有add_normal的标签,却没有add_line的标签。

我们可以通过汇编代码查看是否被inline,汇编代码中,被内联的函数不会有函数的标签,普通的函数会有函数标签。

码字不易,欢迎大家点赞关注评论,谢谢!


C++训练营

专为校招、社招3年工作经验的同学打造的1V1 C++训练营,量身定制学习计划、每日代码review,简历优化,面试辅导,已帮助多名学员获得offer!训练营介绍

相关文章:

  • 鸿蒙开发:RelativeContainer 相对布局详解【全套华为认证学习资料分享(考试大纲、培训教材、实验手册等等)】
  • 能源行业标杆:信创系统在智能电网中的3个创新应用案例
  • com.android.tools.r8.CompilationFailedException: Compilation failed to complete
  • lamp平台的应用
  • Towards Precise and Explainable Hardware Trojan Localization at LUT Level
  • L33.【LeetCode笔记】循环队列(数组解法)
  • ASP.NET Core 6 MVC 文件上传
  • 蓝桥杯P1259-奇怪的馈赠 (贪心题解)
  • File文件和目录
  • Milvus JSON数据存储优化方案
  • 宝塔 Linux 计划任务中添加运行项目网站PHP任务-定时任务
  • Kubernetes教程(三)Docker容器命令
  • 【读书笔记·VLSI电路设计方法解密】问题59:数字电路中的可控性和可观测性是什么
  • 【deepseek】辅助思考生物学问题:ICImapping构建遗传图谱gap较大
  • 基于RapidOCR与DeepSeek的智能表格转换技术实践
  • 大模型LoRA微调训练原理是什么?
  • 【Pandas】pandas Series sort_index
  • 使用Python的requests库调用API并处理JSON响应的详细步骤
  • 【C++进阶学习】第一讲——继承(下)---深入挖掘继承的奥秘
  • Android更新时区版本-ianaVersion
  • 从马相伯到谢希德:复旦大学校长传记系列再版首发
  • 体坛联播|王楚钦晋级男单16强,德布劳内曼城主场谢幕
  • 上海电视节发布海报、宣传片:三十而励,光影新程
  • 今晚油价下调,加满一箱油将省9元
  • 《中华人民共和国经济史(1949—1978年)》教材出版发行
  • 荷兰外交大臣费尔德坎普将访华