一些宏语句可能在C ++中有意想不到的结果?

时间:2011-11-13 00:42:53

标签: c++ c macros

哪个宏语句可能会导致意外结果?

#define YEAR_LENGTH   365
#define MONTH_LENGTH  30
 #define DAYCALC(y, m, d) ((y * YEAR_LENGTH) + (m * MONTH_LENGTH) + d)

 int main()
 {
    int x = 5, y = 4 , z = 1;
    cout << DAYCALC(x *3 , y %3 , z) << endl ;
    cout << DAYCALC(x +12 , y  , 300) << endl ;
    cout << DAYCALC(x , 40 - y , 3+z) << endl ;
    cout << DAYCALC(x  , y  , (z+50)) << endl ;
    cout << DAYCALC(x  , y %3 , z) << endl ;
    cout << DAYCALC(4 % x , y++ , z) << endl;
    return 0;
 }

我运行程序非常好,没有任何意外的结果。

是否有一些隐藏的例外情况?

3 个答案:

答案 0 :(得分:8)

您有运营商优先权问题。宏实际上是作为文本复制和粘贴扩展的。

例如:

DAYCALC(x , 40 - y , 3+z)

扩展为:

((40 - y * YEAR_LENGTH) + (x * MONTH_LENGTH) + 3+z)

请注意,由于运算符优先级,40 - y * YEAR_LENGTH不是您想要的。

因此,您需要将()放在宏中的参数周围:

#define DAYCALC(y, m, d)     (((y) * YEAR_LENGTH) + ((m) * MONTH_LENGTH) + (d))

通常,如果宏参数在宏中出现多次,则y++(在上一个语句中)等副作用也将被多次应用。所以要小心。

答案 1 :(得分:7)

你可以通过g ++ -E检查宏扩展后会发生什么。

int main()
{
int x = 5, y = 4 , z = 1;
cout << ((x *3 * 365) + (y %3 * 30) + z) << endl ; 
cout << ((x +12 * 365) + (y * 30) + 300) << endl ;
cout << ((x * 365) + (40 - y * 30) + 3+z) << endl ; //precedence problem
cout << ((x * 365) + (y * 30) + (z+50)) << endl ; 
cout << ((x * 365) + (y %3 * 30) + z) << endl ;
cout << ((4 % x * 365) + (y++ * 30) + z) << endl;
return 0;
}

答案 2 :(得分:2)

您需要更改DAYCALC;

#define DAYCALC(y, m, d) ( ((y) * YEAR_LENGTH) + ((m) * MONTH_LENGTH) + (d) )

这样,如果m是3 + z,那么内部术语将是正确的;

(3+z) * MONTH_LENGTH

不是不正确的;

3 + z*MONTH_LENGTH