I’ve made a fairly minor 2.99.11 release of the unstable libsigc++, just to make use of C++17. Here are some brief descriptions of the C++17 features used in libsigc++ so far:
std::invoke()
std::invoke() invokes a callable object, such as a plain function pointer, pointer to a member method, or a functor, such as a sigc::slot, using one generic syntax. We provide the arguments, and possibly an object intance, in the std::invoke() call. For instance:
- std::invoke(function, 1, 2);
- std::invoke(method, object, 1, 2);
- std::invoke(functor, 1, 2);
By using std::invoke in libsigc++ (and here), I hope that this makes the implementation more generic though it’s not immediately obvious how.
std::apply()
std::apply()Â is a little like std::invoke(), but lets us invoke a callable object with the arguments in a tuple. For instance:
- std::apply(function, std::make_tuple(1, 2));
- std::apply(method, std::make_tuple(object, 1, 2)); // I think.
- std::apply(functor, std::make_tuple(1, 2));
This is useful for libsigc++ template code that uses tuples to concatenate or rearrange sets of variadic parameters. It lets us replace several slightly less generic helper functions that just called a functor with a tuple unpacked as variadic arguments via a provided std::index_sequence. Receiving variadic parameters into a tuple is normal, so this complements that well, letting us use some of those parameters in calls to other functions.
constexpr if
I also used constexpr if to avoid the need for a helper template. Before C++17, templated code could sometimes need to use another helper template to perform different operations depending on the type, sometimes where some operations won’t even compile with the other type. The different operations would then be isolated in different template specializations of the helper template, and the main template could just have one generic call to that helper template.
But constexpr if lets you put it all in the main template. For instance, it let us remove the little with_type<> template, which only existed so we could call a functor for some types, but not for others.
I also updated the tuple-utils from my murrayc-tuple-utils project, including several uses of constexpr (I also used the new C++17 nested namespace syntax when doing that).