Доступ к общим членам структуры в std::variant

Мне трудно понять, как использовать std::variant в C++17. Учитывая две структуры A и B и std::vector<std::variant<A,B>> vs, я хотел бы:

  • Обратитесь к общему члену структуры, например. n;
  • Вызов общей функции, например. fun() или add().
#include <iostream>
#include <variant>
#include <vector>

struct A {
    int n;
    void fun() { std::cout << "fun\n"; }
    int add(int m) { return n+m; }
};
struct B {
    int n;
    void fun() { std::cout << "fun\n"; }
    int add(int m) { return n+m; }
};

int main() {
    std::vector<std::variant<A,B>> vs;
    vs.push_back(A{10,11});
    vs.push_back(B{20,22});
    
    // How to refer to struct members without using std::get<v.index()>(v)?
    for (auto && v : vs) {
         // 1. How to refer to v.n?
         // 2. How to call v.fun()?
         // 3. How to call v.add() with input parameter m?
    }
}

Мне сказали использовать std::visit, но я слишком туп, чтобы понять, как это работает. Может ли кто-нибудь показать здесь простой пример?


person WDC    schedule 31.08.2020    source источник
comment
Вы смотрели cppreference?   -  person Barry    schedule 31.08.2020
comment
Пример на en.cppreference.com/w/cpp/utility/variant/ посетить помочь? Какие его части смущают вас?   -  person user975989    schedule 31.08.2020
comment
Спасибо вам всем. Кажется, я не понимаю лямбда-выражения...   -  person WDC    schedule 31.08.2020


Ответы (1)


Используйте std::visit с лямбда-выражением с параметром auto&& для доступа к элементам, общим для всех типов вариантов. В вашем примере:

for (auto&& v : vs) {
    std::visit([&](auto&& x){
        std::cout << x.n << x.add(1);
        x.fun();
    }, v);
}
person AtnNn    schedule 31.08.2020
comment
Почему &&, если мы не собираемся от него уходить? - person Vorac; 03.08.2021