目录
一、什么是内联函数?
1、宏函数
#define ADD(x, y) ((x) + (y))
我们先从宏函数说起。当普通函数被多次调用时,就会占用大量的栈帧,但如果函数简单且常用,那么我们可以直接用宏定义函数。因为系统在调用宏函数时不会消耗栈帧,因此用宏函数就可以防止因函数被调用过多而导致爆栈。
但是,宏函数却不会给你进行类型检查;而且由于是预处理指令,所以也不能调试。这就是宏函数的缺点。
2、内联函数的含义
为了解决 C 语言宏函数的这个问题,C++ 发明的内联函数这个东西。它不仅可以像宏函数那样不消耗栈帧,而且还可以进行类型检查,岂不美哉?
3、内联函数的特点
这里我们先说说宏函数的特点。因为宏定义是预处理,程序在预编译是直接把代码中调用宏函数的部分直接换成宏函数的代码,而不像普通函数那样在调用时需额外消耗栈帧。而且,虽然内联函数的声明进入了符号表,但由于内联函数的定义是不会进入符号表的,因此内联函数的地址是一个无效地址。
其实内联函数也和宏函数相似。只不过因为在符号表中的地址是个无效地址,所以就不要指望程序在链接源文件时可以找到它了。因此当我们想写内联函数时,务必要把它和声明放在同一个文件中。
反例:
/*add.h*/
inline int add(int x, int y);
/*add.c*/
inline int add(int x, int y)
{
return x + y;
}
/*Test.c*/
#include "add.h" /*展开 add 头文件*/
#include <iostream>
using namespace std;
int main()
{
int ret = add(3, 5);
return 0;
}
链接错误:main.obj : error LNK2019: 无法解析的外部符号 "void __cdecl f(int)" (f@@YAXH@Z),该符号在函数 _main 中被引用
由于内联函数在符号表中的地址是无效的,因此当程序识别到调用 add 部分时就会找不到 add 的地址,因此就会报错。
4、内联函数的语法
语法和普通的函数一样,但要在返回类型前面加上 inline。
inline int Add(int x, int y);
inline int Add(int x, int y)
{
return x + y;
}
二、内联函数的优缺点
1、优点
内联函数的优点非常明显,它和宏函数的优点非常接近,即在调用时不用消耗栈帧,节省空间的消耗。
2、缺点
但是,内联函数的缺点也十分明显。由于它会直接把函数的代码换到调用它的地方,因此当调用次数多且内联函数的代码过长时,就会大大增加 .exe 文件的大小。
因此,当内联函数的代码量过多时,编译器会自动把它看成普通函数。
三、使用建议
因此,当我们要执行的操作重复次数多且简单时,才可以用内联函数把它封装起来。