c – 预处理器宏作为其他宏的参数

以下C代码编译并作为程序员用于GCC(4.0.4)
#define FOO(x,y,z) ((x)*(y)*(z))
#define BAR(x) FOO(x,1)
#define BAZ 3,7

int main()
{
    return BAR(BAZ); /* interpreted as return ((3)*(7)*(1)); */
}

但是,这些宏会导致Microsoft Visual C Express 2010出现错误

main.cpp(7): warning C4003: not enough actual parameters for macro ‘FOO’
main.cpp(7): error C2059: Syntax error : ‘)’

问题似乎是,Microsoft编译器在内部处理BAR宏时,不会将BAZ宏扩展到可用作宏FOO的两个单独参数的参数.

根据标准,哪个编译器正确处理情况?

解决方法

根据ISO / IEC 14882:2003(C Stardard)的16.3.4进行宏扩展如下:

>宏调用被替换为宏的替换列表(正文),其中每个参数名称(除非受#或##影响)都被宏调用中指定的相应参数的完整宏扩展所替代.
>重新扫描步骤1的结果.如果在其中有更多的宏调用(除了已经扩展获取正在考虑的文本之外),它们将按照相同的过程进行递归扩展.

您指定的代码的步骤顺序如下:

> BAR(BAZ)
> FOO(3,7,1)
>((3)*(7)*(1))

所以GCC是对的,VC不是.但VC抱怨的错误是FOO有3个参数,而BAR只指定其中2个参数. VC显然试图尽快抓住错误,并且在它中有一点太远.

相关文章

/** C+⬑ * 默认成员函数 原来C++类中,有6个默认成员函数: 构造函数 析构函数 拷贝...
#pragma once // 1. 设计一个不能被拷贝的类/* 解析:拷贝只会放生在两个场景中:拷贝构造函数以及赋值运...
C类型转换 C语言:显式和隐式类型转换 隐式类型转化:编译器在编译阶段自动进行,能转就转,不能转就编译...
//异常的概念/*抛出异常后必须要捕获,否则终止程序(到最外层后会交给main管理,main的行为就是终止) try...
#pragma once /*Smart pointer 智能指针;灵巧指针 智能指针三大件//1.RAII//2.像指针一样使用//3.拷贝问...
目录<future>future模板类成员函数:promise类promise的使用例程:packaged_task模板类例程...