C++ 宏和 lambda 捕获

c++ macro and lambda capture

本文关键字:捕获 lambda 宏和 C++      更新时间:2023-10-16

当我在具有多个捕获的宏中使用lambda时,我遇到以下错误之一(Visual Studio 2017(:

Error C2143 syntax error: missing ']' before ';' Error C2958 the left bracket '['

如何避免此错误?

代码示例:

#include <functional>
#include <iostream>
#define MYMACRO(lambda) lambda
int main()
{
    int a = 13;
    int b = 37;
    auto lambda = MYMACRO([a, b]() { std::cout << a << b << std::endl; });
    lambda();
    return 0;
}

GCC 给出了稍微友好的错误消息:

10:73:错误:宏"MYMACRO"传递了 2 个参数,但只占用 1 个参数

lambda 声明中的逗号被解释为分隔宏的参数。您需要将表达式括在括号中:

#include <functional>
#include <iostream>
#define MYMACRO(lambda) lambda
int main()
{
    int a = 13;
    int b = 37;
    auto lambda = MYMACRO(([a, b]() { std::cout << a << b << std::endl; }));
    lambda();
    return 0;
}

Visual Studio 会引发警告,然后忽略虚假的宏参数,因此您的代码等效于:

auto lambda = MYMACRO([a);

这使得错误消息更容易理解。见 https://learn.microsoft.com/en-gb/cpp/error-messages/compiler-warnings/compiler-warning-level-1-c4002

正如其他地方所指出的,实际的错误是编写的MYMACRO只需要一个参数。除非用括号或引号括起来,否则逗号用于描述宏调用的单独参数。

一个 lambda 在捕获括号中可能有多个逗号,因此如果你想让调用语法与程序中的语法相同,MYMACRO实际上需要能够处理变量参数。从 C++11 开始,这是可能的,它增加了对可变参数宏的支持。

Visual Studio 2017 支持可变参数宏语法。因此,您可以将宏更改为:

#define MYMACRO(...) __VA_ARGS__

请注意,变量参数只能出现在宏参数规范的末尾。