使用带有可变参数模板结构的std :: visit

Pau*_*ese 12 c++ c++17

我试图理解以下来自http://en.cppreference.com/w/cpp/utility/variant/visit的例子.

#include <iomanip>
#include <iostream>
#include <string>
#include <type_traits>
#include <variant>
#include <vector>


using var_t = std::variant<int, long, double, std::string>;

template<class... Ts> struct overloaded : Ts... { using Ts::operator()...; };
// what is this declaration imply???
template<class... Ts> overloaded(Ts...) -> overloaded<Ts...>;

int main() {
    std::vector<var_t> vec = {10, 15l, 1.5, "hello"};


    for (auto& v: vec) {
        std::visit(overloaded {
            [](auto arg) { std::cout << arg << '\n'; },
            [](double arg) { std::cout << std::fixed << arg << '\n'; },
            [](const std::string& arg) { std::cout << std::quoted(arg) << '\n'; },
        }, v);
    }
}
Run Code Online (Sandbox Code Playgroud)

有人可以解释这个重载的结构是如何工作的吗?特别是我不明白的是以下声明.

template<class... Ts> overloaded(Ts...) -> overloaded<Ts...>;
Run Code Online (Sandbox Code Playgroud)

如果没有此声明,编译器将发出以下错误消息.

main.cpp: In function 'int main()':
main.cpp:26:9: error: class template argument deduction failed:
         }, v);
         ^
main.cpp:26: confused by earlier errors, bailing out
Run Code Online (Sandbox Code Playgroud)

目的:学习

sky*_*ack 11

有人可以解释这个重载的结构是如何工作的吗?特别是我不明白的是以下声明.

template<class... Ts> overloaded(Ts...) -> overloaded<Ts...>;
Run Code Online (Sandbox Code Playgroud)

这是用户定义的演绎指南(链接到工作草案).
它是该标准的最新版本引入的语言的一个特征以及类模板参数推导.有关更多详细信息和更加用户友好的解释,请参见此处.
这不是一个正确的解释,但为了简单起见,您可以将其视为一个提示,您可以指定从给定构造函数的一组参数中推导出模板参数.


作为旁注,我在这里找到了一个非常清楚的例子,值得复制它:

template<typename T>
struct Thingy { T t; };

Thingy(const char *) -> Thingy<std::string>;

// ...

Thingy thing{"A String"}; // thing.t is a `std::string`.
Run Code Online (Sandbox Code Playgroud)

积分适用于@NicolBolas,这是SO的活跃用户.不幸的是,我找不到这个例子的答案.

  • 应该注意的是`overload'是有效的,因为C++ 17中的聚合初始化被扩展为能够初始化基类子对象以及成员.此外,它是"Nicol Bolas",[世界暴君](http://magic.wizards.com/en/story/planeswalkers/nicol-bolas),而不是尼古拉斯. (3认同)