宏与内联函数

第一部分:宏

为什么要使用宏呢?

因为函数的调用必须要将程序执行的顺序转移到函数所存放在内存中的某个地址,将函数的程序内容执行完后,再返回到转去执行该函数前的地方。这种转移操作要求在转去执行前要保存现场并记忆执行的地址,转回后要恢复现场,并按原来保存地址继续执行。因此,函数调用要有一定的时间和空间方面的开销,于是将影响其效率。而宏只是在预处理的地方把代码展开,不需要额外的空间和时间方面的开销,所以调用一个宏比调用一个函数更有效率。

但是宏也有很多的不尽人意的地方。

在C语言中:

1、宏容易出现一些边界性的问题,产生二义性;

在C++中:

2、宏又不可以调用C++类中的私有或者受保护的成员;

我们举个例子:

#define square(x) (x*x)

我们用一个数字去调用它,如square(5),这样看上去没有什么错误,结果返回25,显然是正确的,但是如果我们用squre(5+5)去调用的话,我们期望的结果是100,而宏的调用结果是(5+5*5+5),结果是35,这显然不是我们要得到的结果。避免这些错误的方法,一是给宏的参数都加上括号。

#define square(x) ((x)*(x))

说明:宏在调用的地方,仅仅是简单的代码替换,所以参数要用括号括起来,不会出现函数调用那种压栈、出栈时的时间和空间的开销,执行效率更高。

第二部分:内联函数

从上面的阐述,可以看到宏有一些难以避免的问题,对于不能访问C++类中私有或者受保护的成员,我们应该如何解决呢?

内联函数是代码被插入到调用者代码处的函数。如同 #define 宏,内联函数通过避免被调用的开销来提高执行效率,尤其是它能够通过调用(“过程化集成”)被编译器优化。

内联函数和宏很类似,而区别在于,宏是由预处理器对宏进行替代,而内联函数是通过编译器控制来实现的。而且内联函数是真正的函数,只是在需要用到的时候,内联函数像宏一样的展开,所以取消了函数的参数压栈,减少了调用的开销。你可以像调用函数一样来调用内联函数,而不必担心会产生于处理宏的一些问题。

内联函数工作原理解释:

对于任何内联函数,编译器在符号表里放入函数的声明(包括名字、参数类型、返回值类型)。

如果编译器没有发现内联函数存在错误,那么该函数的代码也被放入符号表里。

在调用一个内联函数时,编译器首先检查调用是否正确(进行类型安全检查,或者进行自动类型转换,当然对所有的函数都一样)。

如果正确,内联函数的代码就会直接替换函数调用,于是省去了函数调用的开销。

这个过程与预处理有显著的不同,因为预处理器不能进行类型安全检查,或者进行自动类型转换。

假如内联函数是成员函数,对象的地址(this)会被放在合适的地方,这也是预处理器办不到的。

声明内联函数看上去和普通函数非常相似:

void f(int i, char c);

当你定义一个内联函数时,在函数定义前加上 inline 关键字,并且将定义放入头文件:

inline void f(int i, char c){

// ...

}

内联函数必须是和函数体申明在一起,才有效。

像这样的声明inline function(int i)是没有效果的,编译器只是把函数作为普通的函数申明,我们必须定义函数体。

inline int function(int i) { return i*i; }

这样我们才算定义了一个内联函数。我们可以把它作为一般的函数一样调用。但是执行速度确比一般函数的执行速度要快。

当然,内联函数也有一定的局限性。就是函数中的执行代码不能太多了,如果,内联函数的函数体过大,一般的编译器会放弃内联方式,而采用普通的方式调用函数。这样,内联函数就和普通函数执行效率一样了。

有上面的两者的特性,我们可以用内联函数完全取代预处理宏。

第三部分:总结

inline函数的优点与缺点——

优点:

1)inline定义的内联函数,函数代码被放入符号表中,在使用时进行替换(像宏一样展开),效率很高。

2)类的内联函数也是函数。编绎器在调用一个内联函数,首先会检查参数问题,保证调用正确,像对待真正函数一样,消除了隐患及局限性。

3)inline可以作为类的成员函数,可以使用所在类的保护成员及私有成员。

缺点:

内联函数以复制为代价,活动产生开销。

1)如果函数的代码较长,使用内联将消耗过多内存 , 这种情况编译器可能会自动把它作为非内联函数处理。

2)如果函数体内有循环,那么执行函数代码时间比调用开销大。

inline与宏的区别

区别如下:

1)内联在编绎时展开,宏在预编译时展开。 展开的时间不同。

2)编译内联函数可以嵌入到目标代码,宏只是简单文本替换。

3)内联会做类型,语法检查,而宏不具这样功能。

4)宏不是函数,inline函数是函数。

5)宏定义小心处理宏参数(一般参数要括号起来),否则易出现二义性,而内联定义不会出现。

时间: 2024-12-28 22:06:02

宏与内联函数的相关文章

C/C++之宏、内联函数和普通函数的区别

内联函数的执行过程与带参数宏定义很相似,但参数的处理不同.带参数的宏定义并不对参数进行运算,而是直接替换:内联函数首先是函数,这就意味着函数的很多性质都适用于内联函数,即内联函数先把参数表达式进行运算求值,然后把表达式的值传递给形式参数. 内联函数与带参数宏定义的另一个区别是,内联函数的参数类型和返回值类型在声明中都有明确的指定:而带参数宏定义的参数没有类型的概念,只有在宏展开以后,才由编译器检查语法,这就存在很多的安全隐患. 使用内联函数时,应注意以下问题:     1)内联函数的定义性声明应

宏与内联函数的差异探究----自定义MIN函数引发的错误反省

在C++编程中,函数(包括内联函数)一般都是小写,而宏定义的"函数"(带参数的宏)往往采用大写. 上面这句话,看似稀松平常,但是不遵循这句话却容易导致意想不到的错误!今天就记录一个典型案例: 由于内联函数和宏十分相似,都是在程序运行之前进行的,都是用函数体取代表达式,都可以规避函数调用带来的开销从而提高效率,因此很容易模糊二者的本质区别,以至于忘记本文开头的话.这不,今天我就这么做了.这样做固然不符合编程的规范,然而并非一定会导致错误,除非内联函数名和带参数的宏重名,这时如果函数形参没

C99语法之可变参宏和内联函数

可变参宏: 1 #include<stdio.h> 2 #include<stdlib.h> 3 4 #define MYPRINT(...) printf(__VA_ARGS__) 5 6 int main(int argc, char **argv) 7 { 8 MYPRINT("%d,%s", 10, "hello china"); 9 getchar(); 10 return 0; 11 } 使用 ... 来指明多参,使用宏 __A_

C语言(C++)宏、内联函数、函数的区别和使用情形总结

(1)参数传递:宏可以很方便的继承之前代码的变量,函数,内联函数都要靠传参和全局变量 (2)代码生成 宏和内联函数生成时候会替换,没有函数调用时的压栈,因此执行效率会比函数高一些,没有栈溢出的风险,但会生成更多的代码占用更多空间. (3)函数和内联函数要改变输入的参数必须用引用或指针 关于宏 #define STR(str) #str 表示把str加上"" #define STR(str) x##str 表示把str连接上x 已经特殊的符号不能作为宏的参数.

深入探讨 内联函数和宏定义的区别

内联函数的执行过程与带参数宏定义很相似,但参数的处理不同.带参数的宏定义并不对参数进行运算,而是直接替换:内联函数首先是函数,这就意味着函数的很多性质都适用于内联函数,即内联函数先把参数表达式进行运算求值,然后把表达式的值传递给形式参数. 内联函数与带参数宏定义的另一个区别是,内联函数的参数类型和返回值类型在声明中都有明确的指定:而带参数宏定义的参数没有类型的概念,只有在宏展开以后,才由编译器检查语法,这就存在很多的安全隐患. 使用内联函数时,应注意以下问题: 1)内联函数的定义性声明应该出现在

【转】inline内联函数

技术类笔试题50%都会问宏与inline的区别,自己去找找看? 1)宏替换发生在预编译 2)宏函数(如果可以这么叫的话)替换时不会检查参数,inline函数会检查 3)宏一定会发生替换,inline貌似不是强制的,编译器想不替换也没关系 4)宏替换时存在着一些不可避免的陷阱(参见C Traps and Pitfalls),例如传参时如果传了a++之类的可能会出错,inline就比较安全了. 宏有副作用,比如MAX(x++,y++) inline会不同, 慎用内联内联能提高函数的执行效率,为什么不

嵌入式C语言自我修养 10:内联函数探究

10.1 属性声明:noinline & always_inline 这一节,接着讲 attribute 属性声明,attribute可以说是 GNU C 最大的特色.我们接下来继续讲一下跟内联函数相关的两个属性:noinline 和 always_inline.这两个属性的用途是告诉编译器:编译时,对我们指定的函数内联展开或不展开.它们的使用方法如下. static inline __attribute__((noinline)) int func(); static inline __att

C++ inline内联函数

内联函数 函数调用是有时间和空间开销的.程序在执行一个函数之前需要将实参.局部变量.返回地址以及若干寄存器都压入栈中,然后才能执行函数体中的代码:执行完之后,还要将之前压入栈中的数据都出栈,才能接着执行函数调用位置以后的代码. 如果函数体代码比较多,需要较长的执行时间,那么函数调用机制占用的时间可以忽略:如果函数只有一两句语句,那么大部分的时间都会花费在函数调用机制上,这种时间开销就不容忽视. 为了消除函数调用的时空开销,C++在编译时将函数调用处用函数体替换,即内联函数. 注意:要在函数定义出

宏定义与内联函数

1.宏定义的规则和使用解析(1)宏定义的解析规则就是:在预处理阶段由预处理器进行替换,这个替换是原封不动的替换.(2)宏定义替换会递归进行,直到替换出来的值本身不再是一个宏为止.(3)一个正确的宏定义式子本身分为3部分:第一部分是#dedine ,第二部分是宏名 ,剩下的所有为第三部分.(4)宏可以带参数,称为带参宏.带参宏的使用和带参函数非常像,但是使用上有一些差异.在定义带参宏时,每一个参数在宏体中引用时都必须加括号,最后整体再加括号,括号缺一不可. 宏定义示例1:MAX宏,求2个数中较大的