Boost.Fusion run-time switch

前端 未结 5 1521
感情败类
感情败类 2021-02-13 17:43

I am reading the type of an object from a file:

enum class type_index { ... };
type_index typeidx = read(file_handle, type_index{});

Depending

5条回答
  •  再見小時候
    2021-02-13 18:19

    I like my usual inherited lambdas trick:

    I've written about this before

    • Lambda functions as base classes
    • what is the correct way to handle multiple input command differently in c++? (where it visits members of a boost::variant)

    I believe I've seen Sumant Tambe use it in his more recent cpptruths.com postings.


    Demonstration

    Here's a demo for now. Will add some explanation later.

    The most important trick applied is that I use boost::variant to hide the type code denum for us. But the principle applies even if you keep your own type discrimination logic (just requiring more coding)

    Live On Coliru

    #include 
    #include 
    #include 
    #include 
    
    #include 
    #include 
    
    using namespace boost; // brevity
    
    //////////////////
    // This is the utility part that I had created in earlier answers:
    namespace util {
        template struct visitor_t;
    
        template
        struct visitor_t : F1, visitor_t::type {
            typedef visitor_t type;
            visitor_t(F1 head, Fs...tail) : F1(head), visitor_t::type(tail...) {}
    
            using F1::operator();
            using visitor_t::type::operator();
        };
    
        template struct visitor_t : F, boost::static_visitor {
            typedef visitor_t type;
            visitor_t(F f) : F(f) {}
            using F::operator();
        };
    
        template
        typename visitor_t::type make_visitor(Fs...x) { return {x...}; }
    }
    
    using util::make_visitor;
    
    namespace my_types {
        //////////////////
        // fake types for demo only
        struct A1 {
            std::string data;
        };
    
        struct A2 {
            double data;
        };
    
        struct A3 {
            std::vector data;
        };
    
        // some operations defined on A1,A2...
        template  static inline void serialize(A& ar, A1& a, unsigned) { ar & a.data; } // using boost serialization for brevity
        template  static inline void serialize(A& ar, A2& a, unsigned) { ar & a.data; } // using boost serialization for brevity
        template  static inline void serialize(A& ar, A3& a, unsigned) { ar & a.data; } // using boost serialization for brevity
    
        static inline void display(std::ostream& os, A3 const& a3) { os << "display A3: " << a3.data.size() << " elements\n"; }
        template  static inline void display(std::ostream& os, T const& an) { os << "display A1 or A2: " << an.data << "\n"; }
    
        //////////////////
        // our variant logic
        using AnyA = variant;
    
        //////////////////
        // test data setup
        AnyA generate() { // generate a random A1,A2...
            switch (rand()%3) {
                case 0: return A1{ "data is a string here" };
                case 1: return A2{ 42 };
                case 2: return A3{ { 1,2,3,4,5,6,7,8,9,10 } };
                default: throw std::invalid_argument("rand");
            }
        }
    
    }
    
    using my_types::AnyA;
    
    void write_archive(std::string const& fname) // write a test archive of 10 random AnyA
    {
        std::vector As;
        std::generate_n(back_inserter(As), 10, my_types::generate);
    
        std::ofstream ofs(fname, std::ios::binary);
        archive::text_oarchive oa(ofs);
    
        oa << As;
    }
    
    //////////////////
    // logic under test
    template 
    void process_archive(std::string const& fname, F process) // reads a archive of AnyA and calls the processing function on it
    {
        std::ifstream ifs(fname, std::ios::binary);
        archive::text_iarchive ia(ifs);
    
        std::vector As;
        ia >> As;
    
        for(auto& a : As)
            apply_visitor(process, a);
    }
    
    int main() {
        srand(time(0));
    
        write_archive("archive.txt");
    
        // the following is c++11/c++1y lambda shorthand for entirely compiletime
        // generated code for the specific type(s) received
        auto visitor = make_visitor(
            [](my_types::A2& a3) { 
                    std::cout << "Skipping A2 items, just because we can\n";
                    display(std::cout, a3);
                },
            [](auto& other) { 
                    std::cout << "Processing (other)\n";
                    display(std::cout, other);
                }
            );
    
        process_archive("archive.txt", visitor);
    }
    

    Prints

    Processing (other)
    display A3: 10 elements
    Skipping A2 items, just because we can
    display A1 or A2: 42
    Processing (other)
    display A1 or A2: data is a string here
    Processing (other)
    display A3: 10 elements
    Processing (other)
    display A1 or A2: data is a string here
    Processing (other)
    display A1 or A2: data is a string here
    Processing (other)
    display A3: 10 elements
    Processing (other)
    display A1 or A2: data is a string here
    Processing (other)
    display A3: 10 elements
    Processing (other)
    display A3: 10 elements
    

提交回复
热议问题