【GNU笔记】内联函数与宏一样快 An Inline Function is As Fast As a Macro

2024-06-22 03:58

本文主要是介绍【GNU笔记】内联函数与宏一样快 An Inline Function is As Fast As a Macro,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

内联函数与宏一样快 An Inline Function is As Fast As a Macro

通过声明内联函数,你可以指示 GCC 更快地调用该函数。GCC 可以实现这一点的一种方法是将该函数的代码集成到其调用者的代码中。这通过消除函数调用开销使执行速度更快;此外,如果任何实际参数值是常量,则它们的已知值可能允许在编译时进行简化,因此不需要包含所有内联函数的代码。对代码大小的影响是难以预测的;根据具体情况,使用内联函数的目标代码可能更大或更小。你还可以指示 GCC 尝试使用选项 -finline-functions 将所有“足够简单”的函数集成到它们的调用者中。

GCC 实现了声明函数内联的三种不同语义。一种是 -std=gnu89 或者 -fgnu89-inline 或者当gnu_inline 属性出现在所有内联声明中,另一种是 -std=c99 或者 -std=gnu99(没有 -fgnu89-inline),第三种是在编译 C++ 时使用。

要声明一个内联函数,请在其声明中使用inline关键字,如下所示 :

 static inline intinc (int *a){(*a)++;}

如果你正在编写要包含在 ISO C89 程序中的头文件,需要写 __inline__而不是inline. 请参阅替代关键字(Alternate Keywords)。

这三种类型的内联在两种重要情况下表现相似:当 inline 关键字用于 static 函数时,如上面的示例,以及当函数首次声明时不使用 inline 关键字,然后用 inline 定义,如下所示:

 extern int inc (int *a);inline intinc (int *a){(*a)++;}

在这两种常见情况下,程序的行为就像你没有使用inline关键字一样,除了它的速度。

当一个函数同时是 inline 和 static 时,如果对该函数的所有调用都集成到调用者中,并且从不使用函数的地址,则永远不会引用该函数自己的汇编代码。在这种情况下,GCC 不会实际输出函数的汇编代码,除非你指定选项 -fkeep-inline-functions 。 一些调用由于各种原因不能被集成(特别是函数定义之前的调用不能被集成,定义中的递归调用也不能被集成)。如果有一个非集成调用,那么该函数像往常一样被编译为汇编代码。如果程序引用了函数的地址,该函数也必须像往常一样编译,因为它不能被内联。

请注意,函数定义中的某些用法可能使其不适合内联替换。这些用法包​​括:可变参数(varargs)的使用、alloca 的使用、可变大小数据类型(variable sized data types)的使用(请参阅可变长度(Variable Length))、计算 goto (computed goto)的使用(请参阅作为值的标签(Labels as Values))、非局部 goto (nonlocal goto)的使用和嵌套函数(请参阅嵌套函数(Nested Functions))。当标记为 inline 的函数不能被替换时,使用 -Winline 将发出警告,并给出失败的原因。

根据 ISO C++ 的要求,GCC 认为在类中定义的成员函数被标记为内联,即使它们没有使用inline 关键字显式声明。你可以用 -fno-default-inline覆盖它; 请参阅Options Controlling C++ Dialect。

GCC 在不优化时不会内联任何函数,除非你为函数指定了 always_inline 的属性,如下所示:

 /* Prototype.  */inline void foo (const char) __attribute__((always_inline));

本节的其余部分是特定于 GNU C89 内联。

当内联函数不是 static 时,编译器必须假设可能有来自其他源文件的调用;由于一个全局符号在任何程序中只能定义一次,因此不能在其他源文件中定义该函数,因此不能集成其中的调用。因此,非 static 内联函数总是按照通常的方式进行编译。

如果在函数定义中同时指定 inlineextern ,则该定义仅用于内联。在任何情况下,该函数都不会单独编译,即使你显式地引用它的地址也是如此。这样的地址变成了外部引用,就像你只声明了函数,而没有定义它一样。

这种 inlineextern 的组合几乎具有宏的效果。使用它的方法是将函数定义放在带有这些关键字的头文件中,然后将定义的另一个副本(缺少 inline and extern )放在库文件中。头文件中的定义将导致大多数函数调用被内联。如果该函数的任何使用仍然存在,它们将引用库中的单个副本。

Inline - Using the GNU Compiler Collection (GCC)


[参考资料]

5.36 An Inline Function is As Fast As a Macro

本文链接:https://blog.csdn.net/u012028275/article/details/124208461

这篇关于【GNU笔记】内联函数与宏一样快 An Inline Function is As Fast As a Macro的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Oracle的to_date()函数详解

《Oracle的to_date()函数详解》Oracle的to_date()函数用于日期格式转换,需要注意Oracle中不区分大小写的MM和mm格式代码,应使用mi代替分钟,此外,Oracle还支持毫... 目录oracle的to_date()函数一.在使用Oracle的to_date函数来做日期转换二.日

C++11的函数包装器std::function使用示例

《C++11的函数包装器std::function使用示例》C++11引入的std::function是最常用的函数包装器,它可以存储任何可调用对象并提供统一的调用接口,以下是关于函数包装器的详细讲解... 目录一、std::function 的基本用法1. 基本语法二、如何使用 std::function

hdu1171(母函数或多重背包)

题意:把物品分成两份,使得价值最接近 可以用背包,或者是母函数来解,母函数(1 + x^v+x^2v+.....+x^num*v)(1 + x^v+x^2v+.....+x^num*v)(1 + x^v+x^2v+.....+x^num*v) 其中指数为价值,每一项的数目为(该物品数+1)个 代码如下: #include<iostream>#include<algorithm>

【学习笔记】 陈强-机器学习-Python-Ch15 人工神经网络(1)sklearn

系列文章目录 监督学习:参数方法 【学习笔记】 陈强-机器学习-Python-Ch4 线性回归 【学习笔记】 陈强-机器学习-Python-Ch5 逻辑回归 【课后题练习】 陈强-机器学习-Python-Ch5 逻辑回归(SAheart.csv) 【学习笔记】 陈强-机器学习-Python-Ch6 多项逻辑回归 【学习笔记 及 课后题练习】 陈强-机器学习-Python-Ch7 判别分析 【学

系统架构师考试学习笔记第三篇——架构设计高级知识(20)通信系统架构设计理论与实践

本章知识考点:         第20课时主要学习通信系统架构设计的理论和工作中的实践。根据新版考试大纲,本课时知识点会涉及案例分析题(25分),而在历年考试中,案例题对该部分内容的考查并不多,虽在综合知识选择题目中经常考查,但分值也不高。本课时内容侧重于对知识点的记忆和理解,按照以往的出题规律,通信系统架构设计基础知识点多来源于教材内的基础网络设备、网络架构和教材外最新时事热点技术。本课时知识

C++操作符重载实例(独立函数)

C++操作符重载实例,我们把坐标值CVector的加法进行重载,计算c3=c1+c2时,也就是计算x3=x1+x2,y3=y1+y2,今天我们以独立函数的方式重载操作符+(加号),以下是C++代码: c1802.cpp源代码: D:\YcjWork\CppTour>vim c1802.cpp #include <iostream>using namespace std;/*** 以独立函数

论文阅读笔记: Segment Anything

文章目录 Segment Anything摘要引言任务模型数据引擎数据集负责任的人工智能 Segment Anything Model图像编码器提示编码器mask解码器解决歧义损失和训练 Segment Anything 论文地址: https://arxiv.org/abs/2304.02643 代码地址:https://github.com/facebookresear

函数式编程思想

我们经常会用到各种各样的编程思想,例如面向过程、面向对象。不过笔者在该博客简单介绍一下函数式编程思想. 如果对函数式编程思想进行概括,就是f(x) = na(x) , y=uf(x)…至于其他的编程思想,可能是y=a(x)+b(x)+c(x)…,也有可能是y=f(x)=f(x)/a + f(x)/b+f(x)/c… 面向过程的指令式编程 面向过程,简单理解就是y=a(x)+b(x)+c(x)

数学建模笔记—— 非线性规划

数学建模笔记—— 非线性规划 非线性规划1. 模型原理1.1 非线性规划的标准型1.2 非线性规划求解的Matlab函数 2. 典型例题3. matlab代码求解3.1 例1 一个简单示例3.2 例2 选址问题1. 第一问 线性规划2. 第二问 非线性规划 非线性规划 非线性规划是一种求解目标函数或约束条件中有一个或几个非线性函数的最优化问题的方法。运筹学的一个重要分支。2

【C++学习笔记 20】C++中的智能指针

智能指针的功能 在上一篇笔记提到了在栈和堆上创建变量的区别,使用new关键字创建变量时,需要搭配delete关键字销毁变量。而智能指针的作用就是调用new分配内存时,不必自己去调用delete,甚至不用调用new。 智能指针实际上就是对原始指针的包装。 unique_ptr 最简单的智能指针,是一种作用域指针,意思是当指针超出该作用域时,会自动调用delete。它名为unique的原因是这个