类似函数的宏从括号模板参数列表中"steal"逗号是否合法?
Is it legal for a function-like macro to "steal" commas from a parenthesized template argument list?
我只是感到惊讶,为类函数的宏提供带有两个模板参数的类型会导致编译器报错。
这个(概念上类似)示例代码:
template<typename T> struct foo{};
template<typename T, U> struct bar{};
#define p(x) printf("sizeof(" #x ") = %un", sizeof(x));
int main()
{
p(foo<int>); // works, of course
p(bar<int,int>); // does not work
p((bar<int,int>)); // does not work either
return 0;
}
使GCC(6.2.0)报错macro "p" passed 2 arguments, but takes just 1
.
当然,预处理器是一个做文本替换的预处理器,它不是一个真正的c++编译器,它理解模板或语言的所有其他规则。
也许我要求太多了,期望预处理器识别尖括号,授予…但是至少在规范中明确地提到了括号。
16.3(第10至12段)规定了最外层的圆括号分隔有界的令牌序列。单词"最外层"表明可能还存在预处理器识别的其他(非最外层)括号。
此外,它显式地声明"跳过中间匹配的左括号和右括号"以及"匹配的内括号之间的逗号预处理标记不会分开参数"——这意味着如果我阅读正确,那么至少最后一行应该在我的理解中通过。
我理解错了什么?
p((bar<int,int>))
是具有单个宏参数(bar<int,int>)
的p
宏的有效调用。到目前为止你的理解是正确的。
不幸的是,它的扩展包括sizeof((bar<int,int>))
,而sizeof
不接受双括号类型。
可变宏(c++ 11)在这里作为一种替代方法工作得很好。
#define p(...) printf("sizeof(" #__VA_ARGS__ ") = %un", sizeof(__VA_ARGS__));
相关文章:
- 在提升multi_index容器中,是否定义了"default index"?
- 在C++STL中是否有Polyval(Matlab函数)等价物?
- 检查输入是否不是整数或数字
- 是否可以初始化不可复制类型的成员变量(或基类)
- 在C++中,是否可以基于给定的标识符创建基类的新实例,反之亦然
- 是否可以通过C++扩展强制多个python进程共享同一内存
- 此代码是否违反一个定义规则
- 是否需要删除包含对象的"pair"?
- 是否可以从int转换为enum类类型
- 无论条件是否为true,if总是在c++中执行
- 如何找到大小'x'数组是否完全填充,在C++?
- 检查值是否在集合p1和p2中,但不在p3中
- 是否可以在编译时初始化数组,以便在运行时不会花费时间?
- 检查 std::shared_ptr<> 的当前底层类型是否为 T
- 在c++中检查长方体是否尽可能快地重叠(无迭代)
- GL_SHADERSTORAGE_BUFFER位置是否与其他着色器位置冲突
- 子目录是否继承属性,例如add_definitions,include_directories和父Cmakelist.t
- 标准是否使用多余的大括号(例如 T{{{10}}})定义列表初始化?
- C/C++预处理器是否可以检测一些编译器选项
- 类似函数的宏从括号模板参数列表中"steal"逗号是否合法?