预处理器指令以#开头,用于编译前处理;#include包含头文件,查找系统路径,””优先当前目录;#define定义宏,可定义常量如PI或函数如SQUARE(x),但无类型检查,建议用const和inline替代;条件编译#ifdef/#ifndef/#endif与#pragma once防止头文件重复包含;#undef取消宏定义;指令不加分号,宏替换需注意括号避免优先级错误。

在c++中,预处理器指令是在编译之前由预处理器处理的命令。它们以#开头,不遵循常规的C++语法,也不需要分号结尾。其中最常用的是#include和#define,用于文件包含和宏定义。
#include:包含头文件
#include指令的作用是将指定的头文件内容插入到当前源文件中,使程序可以使用标准库或自定义函数、类、常量等。
有两种写法:
- #include
:用于包含系统头文件(如标准库),编译器会在标准路径中查找。 - #include “header”:用于包含用户自定义头文件,编译器优先在当前目录查找,若未找到再搜索系统路径。
常见用法示例:
立即学习“C++免费学习笔记(深入)”;
#include iostream>
#include “myheader.h”
建议在编写项目时,将功能模块拆分为头文件(.h)和实现文件(.cpp),通过#include引入,提高代码可维护性。
#define:定义宏
#define用于定义宏,可以是常量宏或函数宏。它在预处理阶段进行简单的文本替换,不进行类型检查。
1. 定义常量
#define PI 3.14159
#define MAX_SIZE 100
之后代码中所有出现PI的地方都会被替换为3.14159。这种方式类似于常量,但没有类型安全,推荐使用const double PI = 3.14159;替代。
2. 定义宏函数
#define SQUARE(x) ((x) * (x))
#define MAX(a, b) ((a) > (b) ? (a) : (b))
宏函数在使用时会直接展开表达式。注意括号的重要性,防止运算符优先级问题。例如SQUARE(2 + 3)会被展开为((2 + 3) * (2 + 3)),结果正确;若无外层括号,可能出错。
虽然宏函数效率高(无函数调用开销),但调试困难,且容易因副作用出错,例如SQUARE(++x)会导致x被多次递增。建议优先使用内联函数(inline)替代宏函数。
其他常见预处理器指令
- #ifdef / #ifndef / #endif:条件编译,判断某个宏是否已定义,常用于防止头文件重复包含。
- #pragma once:非标准但广泛支持的头文件防重包含方式,比
#ifndef更简洁。 - #undef:取消宏定义,之后该宏不再生效。
示例:防止头文件重复包含
#ifndef MYHEADER_H
#define MYHEADER_H
// 头文件内容
#endif // MYHEADER_H
注意事项
- 预处理器指令不是C++语句,不加分号。
- 宏替换是纯文本替换,容易引发意外错误,需谨慎使用括号。
- 尽量用
const、constexpr和inline替代#define定义的常量和函数。 - 头文件包含应合理组织,避免循环依赖。
基本上就这些。掌握#include和#define是理解C++编译流程的基础,但在现代C++中应更注重类型安全和可维护性。不复杂但容易忽略细节。