可变参数宏:相同代码的不同输出
Variadic macros: Different output for identical code
我在 CLion 中做了一个项目来测试宏。
我创建了一个宏来计算提供给它的参数的数量。
首先,这个宏给了我有效的输出(0 表示没有 args,1 表示一个 arg,2 表示两个 args ...)。
但是后来,我不知道发生了什么,我的宏不知何故失效了。也许是由于很多 SIG_ILL 失败,当我从它的声明范围之外调用函数时未定义的行为,我在表达式中声明了函数。
我试图像在 C++ 中那样实现方法和 new
运算符,但是通过 C 预处理器,这最终是成功的,直到突然我的计算参数的宏失败了。
它现在给出 1 表示没有 args,1 表示一个 args,2 表示两个 args。这个错误发生在编译时,而 CLion 的特性 "inline current usage" 告诉我这个宏在没有参数的情况下调用时应该扩展为 0。
这是这个项目,很简单main.c:
#include <stdio.h>
#define Macro_argument_20(_1, _2, _3, _4, _5, _6, _7, _8, _9, _10, \
_11, _12, _13, _14, _15, _16, _17, _18, _19, _20, ...) _20
#define Macro_argumentsAmount(ARGS...) \
Macro_argument_20("dummy", ##ARGS, 18, 17, 16, 15, 14, 13, 12, 11,10,\
9, 8, 7, 6, 5, 4, 3, 2, 1, 0)
int main() {
printf("%i", Macro_argumentsAmount());
return 0;
}
我不知道并创建了一个新项目以查看会发生什么,所以我 copy/pasted 以前的 CMakeLists.txt 和代码并获得了预期的输出。
所以现在我有两个相同的项目,但输出不同。
有谁知道怎么了?
更新:
仍然在通过
检查源代码的预处理版本时
gcc -E main.c -std=c17 -fplan9-extensions -Werror -Wno-implicit -
Wno-error=variadic-macros -Wno-error=implicit-fallthrough
我在两个项目中都看到了以下主要内容(尽管其中一个打印了 0):
# 9 "main.c"
int main() {
printf("%i", 1);
return 0;
}
当您将 -std=c17
选项添加到编译器时,您禁用了 ##ARGS
的 GCC 逗号吞噬扩展。
改为使用 -std=gnu17
以启用 GNU 扩展以及 c2017 功能。
我在 CLion 中做了一个项目来测试宏。 我创建了一个宏来计算提供给它的参数的数量。 首先,这个宏给了我有效的输出(0 表示没有 args,1 表示一个 arg,2 表示两个 args ...)。 但是后来,我不知道发生了什么,我的宏不知何故失效了。也许是由于很多 SIG_ILL 失败,当我从它的声明范围之外调用函数时未定义的行为,我在表达式中声明了函数。
我试图像在 C++ 中那样实现方法和 new
运算符,但是通过 C 预处理器,这最终是成功的,直到突然我的计算参数的宏失败了。
它现在给出 1 表示没有 args,1 表示一个 args,2 表示两个 args。这个错误发生在编译时,而 CLion 的特性 "inline current usage" 告诉我这个宏在没有参数的情况下调用时应该扩展为 0。
这是这个项目,很简单main.c:
#include <stdio.h>
#define Macro_argument_20(_1, _2, _3, _4, _5, _6, _7, _8, _9, _10, \
_11, _12, _13, _14, _15, _16, _17, _18, _19, _20, ...) _20
#define Macro_argumentsAmount(ARGS...) \
Macro_argument_20("dummy", ##ARGS, 18, 17, 16, 15, 14, 13, 12, 11,10,\
9, 8, 7, 6, 5, 4, 3, 2, 1, 0)
int main() {
printf("%i", Macro_argumentsAmount());
return 0;
}
我不知道并创建了一个新项目以查看会发生什么,所以我 copy/pasted 以前的 CMakeLists.txt 和代码并获得了预期的输出。
所以现在我有两个相同的项目,但输出不同。 有谁知道怎么了?
更新:
仍然在通过
检查源代码的预处理版本时gcc -E main.c -std=c17 -fplan9-extensions -Werror -Wno-implicit -
Wno-error=variadic-macros -Wno-error=implicit-fallthrough
我在两个项目中都看到了以下主要内容(尽管其中一个打印了 0):
# 9 "main.c"
int main() {
printf("%i", 1);
return 0;
}
当您将 -std=c17
选项添加到编译器时,您禁用了 ##ARGS
的 GCC 逗号吞噬扩展。
改为使用 -std=gnu17
以启用 GNU 扩展以及 c2017 功能。