宏定义里面为什么要加括号?

在宏定义当中,常常可以看到宏的参数以及整个宏的定义都被小括号包围,就像下面的 MIN、MAX、ABS 宏一样:

上面的图截取自 iOS 的系统库,那为什么它们需要这些括号包围起来呢?

下面假如我们自定义了宏 ceil_div,代码如下:

#define ceil_div(x, y) (x + y - 1) / y

这个宏的本意是将 x 除以 y,然后将得到的结果向上取整。比如 x = 4,y = 3,那么 ceil_div(4, 3) 的值就是2。如果参数仅仅是这些数字,使用起来没有什么问题,cei_div(4, 3) 经过宏扩展之后成为:

(4 + 3 - 1) / 3

这个符合预期。但是假如参数变得复杂,包含了一些运算符,比如 ceil_div(b & c, sizeof(int)),讲过宏扩展成为下面的样子:

(b & c + sizeof(int) -1 ) / sizeof(int)

由于 & 的优先级比算数运算符 + 的优先级低,那么实际上的运算过程是下面这样的:

(b & (c + sizeof(int)) -1 ) / sizeof(int)

而期望的运算结果是先进行 & 运算,后进行 + 运算:

((b & c) + sizeof(int) -1 ) / sizeof(int)

所以,如果对于宏参数 x、y 不加括号,显然在涉及操作符优先级的情况下,会出现错误。那现在对此进行修正,将宏 ceil_div 的参数用括号进行包围:

#define ceil_div(x, y) ((x) + (y) - 1) / (y)

使用这个定义,就能得到上面期望的结果。但是仅仅只是对宏参数加括号,仍然在一些情况下会有问题,比如像下面使用宏 ceil_div:

sizeof ceil_div(1, 2)

宏被扩展之后,会成为下面的情形:

sizeof ((1) + (2) - 1) / (2)

由于 sizeof 的优先级比算术运算符 / 的优先级高,因此实际上是先进行 sizeof 的运算,再进行除法运算,即:

sizeof (((1) + (2) - 1)) / (2)

而期望的结果是先求宏的值,然后再进行 sizeof 运算:

sizeof (((1) + (2) - 1) / (2))

想要得到正确结果的修改也很简单,直接将整个宏定义使用括号包围即可:

#define ceil_div(x, y) (((x) + (y) - 1) / (y))

因此,宏的参数以及宏定义加括号,是为了解决操作符优先级的问题。

  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值