内联函数
定义:在函数定义体前加入关键字inline,使函数成为内联函数
作用: 增加空间消耗换取效率提高,这点与宏一样
内联函数和普通函数相比可以加快程序运行的速度,因为不需要中断调用
关键字inline必须与函数定义体放在一起才能使函数成为内联,仅仅只是将inline放在函数声明前面没有任何实际作用。
下面函数不能成为内联函数:
inline void test();
void test() {
cout << "useless function!" << endl;
}
下面函数为内敛函数:
void test();
inline void test() {
cout << "useless function!" << endl;
}
#include <iostream>
using namespace std;
class One {
private:
int age;
double len;
inline int max(int x, int y) {
return (x > y) ? x : y;
}
};
inline int min(int x, int y) {
return (x < y) ? x : y;
}
int main() {
cout << "Max(5, 6):" << max(5, 6) << endl;
cout << "Min(5, 6):" << min(5, 6) << endl;
return 0;
}
Max(5, 6):6
Min(5, 6):5
特点:
- 函数体代码量应该较少,一般都是小函数,最好控制在10行以内;
- 内联函数不允许使用循环语句和开关语句;
- 内联函数的定义必须出现内联函数第一次调用之前;
- 在类结构中定义的函数就是类联函数
适用情况:
- 一个函数不断被重复调用
- 函数只有简单几行,且函数内不包括for、while、switch语句
内敛函数运行机制:当函数被声明为内敛函数以后,编译器会将内敛函数展开,不是按照通常函数调用机制进行调用。
内联函数的优缺点:
- 优点:函数体积小,代码运行高效。
- 缺点:滥用内联函数不易控制内存,很容易导致内存增加,这是典型的以空间换时间的函数
宏
作用:
- 定义常量
#define NUM 100
- 条件编译
条件编译的作用主要体现在两个方面:头文件的包含和代码的选择编译
在构建工程的时候, 一般通过#include来包含其他头文件。#include 的处理过程就是将被包含的文件“复制粘贴”到当前文件中, 如果直接或者间接地重复包含了同一个头文件,可能就会导致重复定义等问题。此时可以使用宏定义来避免这一类事情的发生。(也可以使用 #pragma once 来实现该功能, 但此命令跨编译器支持不够好)
#ifndef _HEADER_H_
// some codes
#endif // _HEADER_H_
第二种就是有选择性的编译代码。比如在开发的过程中,常常需要打印一些信息来检查代码运行的正确性。但在最终交付的时候又不需要这些冗余的代码。此时就可以通过宏定义来作为编译的开关, 避免在最终的编译过程中产生不必要的代码。
#ifdef _DEBUG
// run debug codes
#else
// run release codes
#endif // _DEBUG
- 定义函数
一些简单的函数也可以使用宏来实现。因为宏是做简单的替换,并不涉及编译,因此宏展开的函数并不会在代码中产生函数的调用开销, 非常简单高效。值得注意的是,尽量给”参数“加上括号, 防止替换的时候产生歧义,造成错误。同时利用系统定义的可变参数宏, 也能很简洁地实现一些函数,大量简化代码。
#define MIN(a, b) ((a) < (b) ? (a):(b))
#define LOG( format, ... ) printf( format, __VA_ARGS__ )
4.字符串处理
宏有两个特殊的符号 # 和 ##。 前者是将输入的参数字符串化, 即在参数的两端加上双引号; 而后者是做字符串拼接的。
#define STR(a) #a
#define STR_CONTACT(a, b) a##b
宏的展开
宏定义中可以再次添加已经定义过的宏,因此可以实现宏的嵌套。嵌套后的宏定义如何展开,有着一定的规则:
- 实参带入宏文本后,实参之前或者之后遇到#或者##, 均不对其展开。
- 宏展开后遇到“自己”, 则不展开,防止递归展开。
- 带参数的宏,在使用的时候必须加上括号和参数,否则会当做普通的字符,不做展开。
- 最外层的括号内的逗号视为参数分隔符,最内层暂时不视为参数分隔符。
- 宏展开从内层开始进行。
- 每次展开后都会记住展开的宏,后续不再展开相同的宏。
内联函数与宏的差别
- 内联函数运行时可以调试,而宏定义不可以。
- 编译器会对内联函数的参数类型做安全检查或自动类型转化,宏定义不会。
- 内联函数可以访问类的成员变量,宏定义不能。
- 在类中声名同时定义的成员函数,自动转化为内联函数。
- 内联函数将代码直接插入调用处,减少普通函数调用资源,宏是在代码调用处不加任何验证的简单替换。
- 内联函数是函数,宏不是函数。