inline内联类型函数(inline、__always_inline、noinline)

1. 定义

TODO

2. 接口

#define __inline__ inline
#define __inline inline
#define __always_inline	inline __attribute__((always_inline))
#define  noinline	__attribute__((noinline))

3. 优缺点

对于inline:

优点:省去调用子函数所消耗的时间,代码执行效率更高
缺点:代码的体积增大

适用场景:代码量小调用位置少调用频率高
不适用场景:与适用场景相反,重点是递归函数

4. inline、__always_inline、noinline的区别

inline仅仅是建议编译器内联,但不一定内联。
__always_inline强制内联。
noinline强制不内联。

5. 与宏的区别

  • inline预处理阶段不会内联
  • inline类型的函数可以进行参数有效性的检测
  • inline类型的函数返回值可以强制转换为合适的类型

6. 验证inline的方法

C程序

// inline_test.c
#include <stdio.h>
/* 普通函数*/
static void function_1(void)
{
	printf("static void function_1()\n");
	return;
}
/* inline: 建议内联 */
static inline void function_2(void)
{
	printf("static inline void function_2()\n");
	return;
}
/* __always_inline: 强制内联 */
static __always_inline void function_3(void)
{
	printf("static __always_inline void function_3()\n");
	return;
}
/* noinline: 不允许内联 */
static __attribute__((noinline)) void function_4(void)
{
	printf("static __always_inline void function_4()\n");
	return;
}

int main(int argc, const char *argv[])
{
	printf("functiontion main()\n");

	function_1();
	function_2();
	function_3();
	function_4();

	return 0;
}

编译为汇编

命令:
gcc -S inline_test.c -o inline_test.s

从汇编代码可以看出:
function_3被强制内联;
function_1、function_2、function_4都未被内联。

// inline_test.s
	.file	"inline_test.c"
	.section	.rodata
.LC0:
	.string	"static void function_1()"
	.text
	.type	function_1, @function
function_1:
.LFB0:
	.cfi_startproc
	pushq	%rbp
	.cfi_def_cfa_offset 16
	movq	%rsp, %rbp
	.cfi_offset 6, -16
	.cfi_def_cfa_register 6
	movl	$.LC0, %edi
	call	puts
	leave
	ret
	.cfi_endproc
.LFE0:
	.size	function_1, .-function_1
	.section	.rodata
	.align 8
.LC1:
	.string	"static inline void function_2()"
	.text
	.type	function_2, @function
function_2:
.LFB1:
	.cfi_startproc
	pushq	%rbp
	.cfi_def_cfa_offset 16
	movq	%rsp, %rbp
	.cfi_offset 6, -16
	.cfi_def_cfa_register 6
	movl	$.LC1, %edi
	call	puts
	leave
	ret
	.cfi_endproc
.LFE1:
	.size	function_2, .-function_2
	.section	.rodata
	.align 8
.LC2:
	.string	"static __always_inline void function_4()"
	.text
	.type	function_4, @function
function_4:
.LFB3:
	.cfi_startproc
	pushq	%rbp
	.cfi_def_cfa_offset 16
	movq	%rsp, %rbp
	.cfi_offset 6, -16
	.cfi_def_cfa_register 6
	movl	$.LC2, %edi
	call	puts
	leave
	ret
	.cfi_endproc
.LFE3:
	.size	function_4, .-function_4
	.section	.rodata
.LC3:
	.string	"functiontion main()"
	.align 8
.LC4:		// static __always_inline, 强制内联
	.string	"static __always_inline void function_3()"
	.text
.globl main
	.type	main, @function
main:
.LFB4:
	.cfi_startproc
	pushq	%rbp
	.cfi_def_cfa_offset 16
	movq	%rsp, %rbp
	.cfi_offset 6, -16
	.cfi_def_cfa_register 6
	subq	$16, %rsp
	movl	%edi, -4(%rbp)
	movq	%rsi, -16(%rbp)
	movl	$.LC3, %edi
	call	puts
	call	function_1  // static, 调用的形式
	call	function_2  // static inline, 建议内联,但不一定会内联
	movl	$.LC4, %edi // function_3()被强制内联于此
	call	puts
	call	function_4  // function_4()强制不内联
	movl	$0, %eax
	leave
	ret
	.cfi_endproc
.LFE4:
	.size	main, .-main
	.ident	"GCC: (Ubuntu/Linaro 4.4.7-8ubuntu1) 4.4.7"
	.section	.note.GNU-stack,"",@progbits
  • 3
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值