2016-06-27 25 views
5

devo spinta tipo di variante definizione:Come semplificare l'azione positiva sulla variante boost?

typedef boost::variant< bool, int, float, double> VariantType; 

Voglio realizzare add/sottrazione/moltiplicazione/azione divide su di esso. Prendi Aggiungi classe per esempio. Il problema è se si aggiunge un nuovo tipo in VariantType, come std :: string, la classe Add deve essere aggiornata con il nuovo tipo.

struct Add : public boost::static_visitor<VariantType> { 
    template <typename T> 
    T operator() (T a, T b) const { 
     return a + b; 
    } 
    float operator() (int a, float b) const { 
     return a + b; 
    } 
    float operator() (float a, int b) const { 
     return a + b; 
    } 
    double operator() (int a, double b) const { 
     return a + b; 
    } 
    double operator() (double a, int b) const { 
     return a + b; 
    } 
    double operator() (float a, double b) const { 
     return a + b; 
    } 
    double operator() (double a, float b) const { 
     return a + b; 
    } 
    VariantType operator() (bool a, int b) const { 
     throw std::invalid_argument("bool and int can't Plus"); 
    } 
    VariantType operator() (bool a, float b) const { 
     throw std::invalid_argument("bool and float can't Plus"); 
    } 
    VariantType operator() (bool a, double b) const { 
     throw std::invalid_argument("bool and double can't Plus"); 
    } 
    VariantType operator() (int a, bool b) const { 
     throw std::invalid_argument("int and bool can't Plus"); 
    } 
    VariantType operator() (float a, bool b) const { 
     throw std::invalid_argument("float and bool can't Plus"); 
    } 
    VariantType operator() (double a, bool b) const { 
     throw std::invalid_argument("double and bool can't Plus"); 
    } 
}; 

utilizzo:

VariantType v1 = 1; 
VariantType v2 = 2.1; 
VariantType v3 = boost::apply_visitor(Add(), v1, v2); 
cout<<boost::get<double>(v3)<<endl; //Print 2.2 

La versione GCC 4.8.2 è; La versione boost è 1.57.0; Come semplicemente aggiungere la classe? Grazie.

risposta

3

Basta fare un funtore polimorfa:

Live On Coliru

#include <boost/variant.hpp> 
#include <iostream> 

using VariantType = boost::variant<int, float, double, bool>; 

struct Add : public boost::static_visitor<VariantType> { 
    template <typename T, typename U> 
    auto operator() (T a, U b) const -> decltype(a+b) { return a + b; } 

    template <typename T> VariantType operator()(bool, T) const { throw std::invalid_argument("Can't to bool"); } 
    template <typename T> VariantType operator()(T, bool) const { throw std::invalid_argument("Can't add bool"); } 
    VariantType      operator()(bool,bool) const { throw std::invalid_argument("Can't add bools"); } 
}; 

int main() { 
    std::cout << std::boolalpha; 

    VariantType specimens[] = { int(42), 3.14f, 3.14, true }; 
    for (auto lhs : specimens) 
     for (auto rhs : specimens) 
     { 
      try { 
       std::cout << lhs << " + " << rhs << " == " << boost::apply_visitor(Add{}, lhs, rhs) << "\n"; 
      } catch(std::exception const& e) { 
       std::cout << lhs << " + " << rhs << " ==> " << e.what() << "\n"; 
      } 
     } 
} 

Stampe:

42 + 42 == 84 
42 + 3.14 == 45.14 
42 + 3.14 == 45.14 
42 + true ==> Can't add bool 
3.14 + 42 == 45.14 
3.14 + 3.14 == 6.28 
3.14 + 3.14 == 6.28 
3.14 + true ==> Can't add bool 
3.14 + 42 == 45.14 
3.14 + 3.14 == 6.28 
3.14 + 3.14 == 6.28 
3.14 + true ==> Can't add bool 
true + 42 ==> Can't to bool 
true + 3.14 ==> Can't to bool 
true + 3.14 ==> Can't to bool 
true + true ==> Can't add bools 
+0

Ho appena aggiunto un esempio dal vivo con molte piccole correzioni/chiarimenti/cc @ DanielSchepler – sehe

+0

Funziona. Grazie – liuzw