Mengakses anggota struct umum di std::varian

Saya kesulitan memahami cara menggunakan std::variant di C++17. Diberikan dua struct A dan B, dan std::vector<std::variant<A,B>> vs, saya ingin:

  • Merujuk ke anggota struct umum, mis. n;
  • Panggil fungsi umum, mis. fun() atau 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?
    }
}

Saya diberitahu untuk menggunakan std::visit, tetapi saya terlalu blak-blakan untuk memahami cara kerjanya. Adakah yang bisa menunjukkan contoh sederhana di sini?


person WDC    schedule 31.08.2020    source sumber
comment
Apakah Anda melihat cppreference?   -  person Barry    schedule 31.08.2020
comment
Apakah contohnya ada di en.cppreference.com/w/cpp/utility/variant/ kunjungi bantuan? Bagian mana yang membuatmu bingung?   -  person user975989    schedule 31.08.2020
comment
Terima kasih semua. Sepertinya saya tidak mengerti ekspresi lambda...   -  person WDC    schedule 31.08.2020


Jawaban (1)


Gunakan std::visit dengan lambda yang memiliki parameter auto&& untuk mengakses anggota yang umum untuk semua jenis varian. Dalam contoh Anda:

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
Kenapa && jika kita tidak berniat beranjak darinya? - person Vorac; 03.08.2021