如何从 std::visit return const&?

How to return const& from std::visit?

我遇到了我认为 std::visitoverloaded 的奇怪情况,这给我一个编译器错误。为了说明我正在尝试做的事情,我有一个使用 std::visit 3 种不同方式的示例。

我的目标 是能够将 std::visit 部分包装在一个函数中,因为在我的用例中它不是一小段代码。我有多个 lambda 重载。我不想在每个使用变体的函数中复制这段代码并希望获得 属性,我想在某种包装器中编写一次,我可以在多个翻译单元中重复使用(方法 3 对我来说是最可取的,然后是方法 2,如果其他都失败了,我会研究一些不太令人满意的东西,比如宏或其他东西。)

谁能给我解释一下:

请考虑这个重现我遇到的问题的最小示例。

最小示例:

#include <iostream>
#include <string>
#include <variant>

template<class... Ts> struct overloaded : Ts... { using Ts::operator()...; };

struct a {
    std::string name;

    std::string const& get_name() const { return name; }
};

struct b {
    std::string name;

    std::string const& get_name() const { return name; }
};

std::string const& visit_get_name(std::variant<a, b> const& data) {
    return std::visit(overloaded{[](auto const& o){return o.get_name();}}, data);
}

struct wrapper {

    std::string const& get_name() const {
        return std::visit(overloaded{[](auto const& o){return o.get_name();}}, data);
    }

    std::variant<a, b> data;
};

int main(int, char**) {
    std::variant<a, b> a_thing{a{"sue"}};
    wrapper a_wrapper{a_thing};

    std::cout << "Approach 1: " << std::visit(overloaded{[](auto const& o){return o.get_name();}}, a_thing);
    std::cout << "Approach 2: " << visit_get_name(a_thing);
    std::cout << "Approach 3: " << a_wrapper.get_name();

    return 0;
}

编译器:

GCC 11.2.0

编译器错误:

example.cpp: In function 'const string& visit_get_name(const std::variant<a, b>&)':
example.cpp:20:22: error: returning reference to temporary [-Werror=return-local-addr]
   20 |     return std::visit(overloaded{[](auto const& o){return o.get_name();}}, data);
      |            ~~~~~~~~~~^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
example.cpp: In member function 'const string& wrapper::get_name() const':
example.cpp:26:26: error: returning reference to temporary [-Werror=return-local-addr]
   26 |         return std::visit(overloaded{[](auto const& o){return o.get_name();}}, data);
      |                ~~~~~~~~~~^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

编译器开关:

-Wall 
-Wextra 
-Wshadow 
-Wnon-virtual-dtor 
-Wold-style-cast 
-Wcast-align 
-Wunused 
-Woverloaded-virtual 
-Wpedantic 
-Wconversion 
-Wsign-conversion 
-Wnull-dereference 
-Wdouble-promotion 
-Wformat=2 
-Werror 
-Wmisleading-indentation 
-Wduplicated-cond 
-Wduplicated-branches 
-Wlogical-op 
-std=gnu++20

调用 lambda(或与此相关的函数)时的默认行为是通过复制获得值 return。由于您通过复制传递给 overloaded return 的 lambda 表达式,绑定对 visit_get_name(或 wrapper::get_name)的 return 类型的引用不是允许,这就是方法 2 和 3 失败的原因。

如果你想通过引用 return,你必须这样说(注意 lambda 的显式 -> auto const & 尾随 return 类型),像这样:

return std::visit(overloaded{[](auto const& o) -> auto const & {
                               return o.get_name();
                             }}, data);

wrapper::get_name 也是如此)。

方法 1 不需要这样做,因为您没有绑定对 returned 内容的引用。

demo.