c++ 中宏备忘

来自:
http://blog.youkuaiyun.com/szlanny/archive/2009/06/12/4264663.aspx


1.宏不要用来定义常量,因为宏变量是没有类型安全的,也没有名字空间约束,会造成名字的污染。

2.宏的展开是一行,所以宏中的注释不能使用//,只能使用/* */。宏的代码也不能gdb跟踪,宏中代码逻辑要尽量简单。

3.宏的参数一般情况下使用时要用()括起来,如

#define MAX(a, b) a /2 > b ? a /2 : b

MAX(3,4)使用没有问题,但MAX(3+4, 4)却有问题,因为宏的参数仅为符号替换。

应用定义为#define MAX(a, b) (a) / 2 > (b) ? (a) /2 : (b)

4.宏的连接符分为#与##

#表示一个符号直接转换为字符串,如

#define CAT(x) "First "#x " Third"
const char * pszStr = CAT(Second); str的内容就是"First Second Third",也就是说#会把其后的符号直接加上双引号。

##符号会连接两个符号,从而产生新的符号(词法层次),例如:

#define NAME( x ) name_##x

char* NAME( szlanny ); 宏被展开后将成为:char* name_szlanny;

5.宏中如有存在if等语句产生的分支,要使用do{}while(0)包起来,如

#define TEST(a ) if ( 0 == a ) dosomething()

如果在下面使用是会存在问题

if ( 1 == b)

TEST(a ):

else

{

dootherthing();

}

那当代码展开之后,宏中的if与外面的else是一起匹配,而不是else与if ( 1 == b)匹配。

所以上述宏要修改为 #define TEST(a ) do {\

if ( 0 == a ) { dosomething(); } }while(0)

6.对于可变参宏,可以使用__VA_ARGS__,GCC支持它,但并非所有的编译器支持,如果你的代码要跨平台,慎用。
#define LOG( format, ... ) printf( format, __VA_ARGS__ )

LOG( "%s %d", str, count );
__VA_ARGS__被自动替换为参数列表。

7.宏不能嵌套使用,如

#define TEST( x ) ( x + TEST( x ) ) , 编译器展开过程中发现第二个TEST,那么就将这个TEST当作一般的符号。

8.当宏的逻辑比较多时,可以考虑宏中使用模板方法来代替宏的逻辑实现。

如果,调用的时候,需要连接的符号,传了字符串。
会报 ,does not give a valid preprocessing token 。
expected initializer before string constant 。
### 定义的作用与使用方法 在C++中,定义是一种预处理指令,用于在编译之前替换代码中的特定符号。它可以通过提高代码的通用性、减少输入错误和便于修改来增强程序的可维护性和易读性[^2]。 #### 1. 基本定义 基本定义通过`#define`指令实现,格式如下: ```cpp #define 名 替换内容 ``` 例如: ```cpp #define MAX 1000 ``` 在此例中,所有出现的`MAX`都会被替换为`1000`。需要注意的是,定义末尾不加分号,并且预处理阶段不会进行语法检查[^2]。 #### 2. 定义的作用域 定义通常写在函数花括号外边,其作用域从定义处开始直到文件结束。如果需要限制的作用范围,可以使用`#undef`命令终止定义的作用域。例如: ```cpp #define PI 3.14 #undef PI ``` 在此之后,`PI`将不再有效[^2]。 #### 3. 带参定义 带参定义允许在中传递参数,类似于函数调用。格式如下: ```cpp #define 名(参数列表) 替换内容 ``` 例如: ```cpp #define SQUARE(x) ((x) * (x)) ``` 在此例中,`SQUARE(5)`会被替换为`((5) * (5))`。需要注意的是,为了防止运算符优先级问题,建议对参数和整个表达式加上括号。 #### 4. 定义嵌套 定义可以嵌套使用,即在一个定义中引用另一个定义。例如: ```cpp #define A 100 #define B (A + 200) ``` 在此例中,`B`会被替换为`(100 + 200)`。 #### 5. 字符串中的使用 需要注意的是,字符串中永远不会包含。例如: ```cpp #define NAME "John" std::cout << "Hello, " NAME; ``` 此代码会输出`Hello, John`,但直接在字符串中写`NAME`则不会被替换[^2]。 #### 6. C++中的替代方案 尽管定义在C语言中广泛使用,但在C++中,建议使用`const`或`constexpr`变量代替定义,特别是在类内或方法内。例如: ```cpp const int MAX = 1000; ``` 这种方式不仅更安全,还能更好地融入C++的类型系统[^1]。 #### 7. 定义与CMake 在使用CMake构建项目时,可以通过`add_compile_definitions`命令定义全局。例如: ```cmake add_compile_definitions(MY_MACRO) ``` 此外,还可以根据不同的生成器和平台提供不同的定义: ```cmake add_compile_definitions($<$<CXX_COMPILER_ID:MSVC>:WINDOWS>) ``` 这使得定义更加灵活,适用于跨平台开发[^3]。 #### 8. 可变参数 C99引入了可变参数C++也支持这种功能。通过`__VA_ARGS__`可以处理可变数量的参数。例如: ```cpp #define DEBUG_PRINT(format, ...) printf(format, __VA_ARGS__) ``` 此可以接受任意数量的参数,并将其传递给`printf`函数[^4]。 ### 示例代码 以下是一个综合示例,展示了如何在C++中使用定义: ```cpp #include <iostream> #define MAX(a, b) ((a) > (b) ? (a) : (b)) #define PI 3.14159 #define DEBUG_PRINT(fmt, ...) std::printf(fmt "\n", __VA_ARGS__) int main() { int x = 10, y = 20; std::cout << "Max: " << MAX(x, y) << std::endl; double radius = 5.0; std::cout << "Area: " << PI * radius * radius << std::endl; DEBUG_PRINT("Debug message: %d", x); return 0; } ```
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值