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

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

#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; } 

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

有一些隐藏的例外吗?

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

例如:

 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++ (在您的上一个语句中)等副作用也将被多次应用。 所以要小心。

你可以检查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; } 

你需要改变DAYCALC;

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

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

 (3+z) * MONTH_LENGTH 

不是不正确的;

 3 + z*MONTH_LENGTH