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
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
).