Binding a generic member function

前端 未结 1 1946
一向
一向 2021-01-07 02:51

Sometimes I need to bind some member functions to its calling object, to treat member functions and non-member functions in the same homogeneous way. For example (The tipica

1条回答
  •  一向
    一向 (楼主)
    2021-01-07 03:39

    To work with std::bind, we need to somehow supply a certain amount of placeholders, depending on the amount of function parameters of the callback. I've described a way here how you can do that, by creating a generator for placeholders:

    template struct placeholder_template {};
    

    By partially specializing std::is_placeholder for the above template, std::bind sees the instantiations placeholder_template as placeholder types. With the usual indices trick, we then expand placeholder_template<0>{}, placeholder<1>{}, ...., placeholder{}, where N is the number of function parameters.

    template
    class callback_list
    {
    public:
        using callback_t = std::function;
    
        //Overload for non-member handlers:
        void add( const callback_t& handler )
        {
            _handlers.push_back( handler );
        }
    
    private:
        //Overload for member handlers:
        template
        void add( CLASS& object_ref , 
                  void(CLASS::*member_function)( Params... ) ,
                  int_sequence )
        {
            using namespace std::placeholders;
    
            _handlers.push_back( std::bind( member_function , 
                                            std::ref( object_ref ) , 
                                            placeholder_template{}...
                                          ) 
                               );
        }
    public:
        template
        void add( CLASS& object_ref , 
                  void(CLASS::*member_function)( Params... ) )
        {
            add( object_ref, member_function,
                 make_int_sequence{} );
        }
        
    
        template
        void operator()( ARGS&&... args )
        {
            for( auto& handler : _handlers )
                handler( std::forward( args )... );
        } 
    
    private:
        std::vector _handlers;
    };
    

    The code for the placeholder generator and the integer sequence, from the other answer:

    template struct int_sequence {};
    
    template struct make_int_sequence
        : make_int_sequence {};
    template struct make_int_sequence<0, Is...>
        : int_sequence {};
    
    template // begin with 0 here!
    struct placeholder_template
    {};
    
    #include 
    #include 
    
    namespace std
    {
        template
        struct is_placeholder< placeholder_template >
            : integral_constant // the one is important
        {};
    }
    

    Using C++14, one can use std::index_sequence_for instead of custom make_int_sequence.


    Side remark: If you want to accept member functions with cv- and ref-qualifiers, you could use a very general "pattern" like

    template
    void add(C& ref, T fun);
    

    and restrict via SFINAE. Here's a trait that lets you deduce the number of parameters from such a function pointer (via tuple_size).

    0 讨论(0)
提交回复
热议问题