C++中inline函数详解_C 语言

本文主要记录了C++中的inline函数,也就是内联函数,主要记录了以下几个问题:

一、C++为什么引入inline函数?

主要目的:用它代替C语言中表达式形式的宏定义来解决程序中函数调用的效率问题。

C语言中的宏定义:#define ExpressionName(var1,var2) (var1+var2)*(var1-var2)这种宏定义,它使用预处理器实现,没有了参数压栈、代码生成等一系列得到操作,因此效率很高。但缺点如下:

仅仅是做预处理器符号表中的简单替换,因此不能进行参数有效性的检测,不能享受C++编译器严格类型检查的好处。
另外,它的返回值也不能被强制转换为可转换的合适类型。
还有,C++引入了类及类的访问控制,这样,如果一个操作或者说一个表达式涉及类的保护成员或者私有成员,这种宏定义就无法实现(因为无法将this指针放在合适的位置上)

二、为什么inline能很好的取代表达式形式的预定义?

inline定义类的内联函数,函数的代码被放入符号表中,在使用时直接进行替换(像宏一样展开)没有了调用的开销,效率也高了。
类的内联函数也是一个真正的函数。编译器在调用一个内联函数的时候,首先进行一系列的检测(参数的类型)
inline函数可以作为某个类的成员函数,这就可以在其中使用该类的保护成员和私有成员。

三、inline函数的使用场合

示例代码:

由上述示例代码可知:A类的2个成员函数都是inline函数,readTest()函数的定义在类内,所以自动转化为inline函数,setTest()函数的定义在类外,所以必须加inline关键字。类的成员常定义成保护和私有的,外界不能直接访问这些成员,所以必须有成员接口函数来访问。这些接口函数被定义成inline函数,会获得比较好的效果。所以,inline函数常用于定义存取函数(代码简短),inline函数的效率比较高。

四、为什么不把所有的函数定义成inline函数?

inline是以代码的膨胀(复制)为待见的,仅仅省去了函数调用的开销,从而提高了函数的执行效率。如果,执行函数体内代码的时间相比于函数调用的开销大,那么效率的收获会很少。另一方面,每一个inline函数的调用都要复制代码,使程序的总代码量增大,消耗更多的内存空间。
所以:

函数的代码量比较大时,使用inline函数会使内存消耗代价较高。
函数体内出现循环,那么执行函数的时间要比函数调用的开销大。
另外,类的构造函数和析构函数容易让人误解成使用inline更有效。要当心构造函数和析构函数可能会隐藏一些行为,如“偷偷地执行基类或成员对象的构造函数和析构函数。”
一个好的编译器会根据函数体,自动取消不适合的inline函数。(说明了,inline不应该出现在类的内部,及函数的声明的部分)

五、inline函数与宏的区别?

inline是在 编译 时展开的,而宏是在 预编译 时展开的。
在编译时,inline函数可以直接嵌套到目标代码里,而宏只是简单地文本替换
inline函数可以完成类型和语句是否正确,而宏不具有这样的功能。
inline函数是函数,而宏不是函数。
宏的定义时,小心参数的处理(一般把参数用括号括起来),否则会引起二义性,而inline函数不用担心二义性。

以上所述就是本文的全部内容了,希望大家能够喜欢。

时间: 2024-08-24 08:01:38

C++中inline函数详解_C 语言的相关文章

从汇编看c++中的多态详解_C 语言

在c++中,当一个类含有虚函数的时候,类就具有了多态性.构造函数的一项重要功能就是初始化vptr指针,这是保证多态性的关键步骤. 构造函数初始化vptr指针 下面是c++源码: class X { private: int i; public: X(int ii) { i = ii; } virtual void set(int ii) {//虚函数 i = ii; } }; int main() { X x(1); } 下面是对应的main函数汇编码: _main PROC ; 16 : in

C/C++字符串函数之复制函数详解_C 语言

突然发现对字符串函数缺乏系统的了解,所以花了一点时间专门整理下,在此记录之,以方便自己及有需要的人使用. C/C++字符串函数的头文件:string.h 复制函数主要有4个,如下: 1.char * strcpy(char* destination,const char * source); 2.char* strncpy(char* destination,const char* source,size_t num); 3.void * memcpy(void* destination,con

C语言中改变目录的相关操作函数详解_C 语言

C语言fchdir()函数:改变当前工作目录头文件: #include <unistd.h> 定义函数: int fchdir(int fd); 函数说明:fchdir()用来将当前的工作目录改变成以参数fd 所指的文件描述词. 返回值:执行成功则返回 0, 失败返回-1, errno 为错误代码. 范例 #include <sys/types.h> #include <sys/stat.h> #include <fcntl.h> #include <

C语言fillpoly函数详解_C 语言

C语言中,fillpoly函数的功能是画一个多边形,今天我们就来学习学习. C语言fillpoly函数:填充一个多边形 函数名:fillpoly 功  能:画并填充一个多边形 头文件:#include <graphics.h> 原  型:fillpoly(int numpoints, int far *polypoints); 参数说明:numpoints 为多边形的边数:far *polypoints 为存储各顶点坐标的数组,每两个一组表示一个顶点的 X 和 Y 坐标. 实例代码: #inc

C++中CSTRINGLIST用法详解_C 语言

CStringList类成员 构造 CStringList 构造一个空的CString对象列表 首/尾访问 GetHead 返回此列表(不能是空的)中头部的元素 GetTail 返回此列表(不能是空的)中尾部的元素 操作 RemoveHead 从列表的头部删除元素 RemoveTail 从列表的尾部删除元素 AddHead 在列表的头部添加一个元素(或者是另一个列表中的所有元素),即产生一个新的头部 AddTail 在列表的尾部添加一个元素(或者是另一个列表中的所有元素),即产生一个新的尾部 R

c++内联函数(inline)使用详解_C 语言

介绍内联函数之前,有必要介绍一下预处理宏.内联函数的功能和预处理宏的功能相似.相信大家都用过预处理宏,我们会经常定义一些宏,如 复制代码 代码如下: #define TABLE_COMP(x) ((x)>0?(x):0) 就定义了一个宏. 为什么要使用宏呢?因为函数的调用必须要将程序执行的顺序转移到函数所存放在内存中的某个地址,将函数的程序内容执行完后,再返回到转去执行该函数前的地方.这种转移操作要求在转去执行前要保存现场并记忆执行的地址,转回后要恢复现场,并按原来保存地址继续执行.因此,函数调

C语言中进程信号集的相关操作函数详解_C 语言

C语言sigismember()函数:测试某个信号是否已加入至信号头文件:#include <signal.h> 定义函数:int sigismember(const sigset_t *set, int signum); 函数说明:sigismember()用来测试参数signum 代表的信号是否已加入至参数set 信号集里. 如果信号集里已有该信号则返回1, 否则返回0. 返回值:信号集已有该信号则返回1, 没有则返回0.如果有错误则返回-1. 错误代码: 1.EFAULT 参数set 指

WM_CLOSE、WM_DESTROY、WM_QUIT及各种消息投递函数详解_C 语言

本文对WM_CLOSE.WM_DESTROY.WM_QUIT及各种消息投递函数的功能及区别做出了分析比对,有助于读者更好的对消息投递函数加以理解.详情如下: 一.WM_CLOSE.WM_DESTROY.WM_QUIT区别 WM_CLOSE:关闭应用程序窗口 WM_DESTROY:关闭应用程序 WM_QUIT:关闭消息循环 只有关闭了消息循环,应用程序的进程才真正退出(在任务管理器里消失). win32应用程序的完整退出过程:点击窗口右上角的关闭按钮,发送WM_CLOSE消息.此消息处理中调用De

C++类静态成员与类静态成员函数详解_C 语言

当将类的某个数据成员声明为static时,该静态数据成员只能被定义一次,而且要被同类的所有对象共享.各个对象都拥有类中每一个普通数据成员的副本,但静态数据成员只有一个实例存在,与定义了多少类对象无关.静态方法就是与该类相关的,是类的一种行为,而不是与该类的实例对象相关. 静态数据成员的用途之一是统计有多少个对象实际存在. 静态数据成员不能在类中初始化,实际上类定义只是在描述对象的蓝图,在其中指定初值是不允许的.也不能在类的构造函数中初始化该成员,因为静态数据成员为类的各个对象共享,否则每次创建一