define 是 C++ 预处理器指令,用于编译前文本替换,无类型安全与作用域限制;适用于条件编译、简单常量、头文件卫士等,但需谨慎处理括号和副作用。

#define 是 C++ 预处理器指令,用于文本替换,不是类型安全的,也不受作用域限制。它在编译前就被展开,所以不参与语法检查、调试困难、容易出错——但仍有实用场景,比如条件编译、简单常量、头文件卫士等。
基本用法:常量与简单函数式宏
定义常量:
```cpp#define PI 3.1415926
#define MAX(a, b) ((a) > (b) ? (a) : (b))
```
注意:
• PI 后面没有分号,否则会把分号也替进去;
• MAX 中每个参数都加了括号,防止运算符优先级问题(如 MAX(x + 1, y * 2));
• 整个表达式也加括号,避免嵌入到更大表达式时出错(如 2 * MAX(3, 4))。
常见陷阱:括号缺失与多次求值
错误写法(没加括号):
立即学习“C++免费学习笔记(深入)”;
```cpp#define SQUARE(x) x * x // 危险!
int a = 5;
int b = SQUARE(a + 1); // 展开为 a + 1 * a + 1 → 5 + 1 * 5 + 1 = 11
```
正确写法:
```cpp#define SQUARE(x) ((x) * (x))
```
另一个经典陷阱是带副作用的参数:
#define MIN(a, b) ((a) int i = 0;
int m = MIN(i++, 5); // i 被递增两次!展开后类似:((i++) ```
这种宏会引发未定义行为,应改用 inline 函数 或 constexpr 函数 替代。
更安全的替代方案(推荐优先使用)
• 常量:用 constexpr int MAX_SIZE = 100; 或 const double PI = 3.14159;
• 函数式逻辑:用 inline constexpr auto max = [](auto a, auto b) { return a > b ? a : b; }; 或普通模板函数
• 条件编译仍需 #ifdef 系列,但可配合 __has_include、__cplusplus 等标准宏增强可移植性
实用技巧:头文件卫士与调试宏
防止重复包含:
```cpp#ifndef MY_HEADER_H
#define MY_HEADER_H
// 头文件内容
#endif
```
调试时开关日志(只在 Debug 模式生效):
```cpp#ifdef DEBUG
#define LOG(msg) std::cout #else
#define LOG(msg) do {} while(0)
#endif
```
注意:do {} while(0) 是空宏的标准写法,保证语法上可接分号、支持 if-else 结构,不会产生意外分支。
基本上就这些。宏不是不能用,而是要清楚它只是“文本粘贴”,没有类型、没有作用域、没有调试信息。能用 const / constexpr / inline 解决的,就别用 #define。









