Generally speaking, unless explicitly allowed, the behavior of a C++ program that tries to take the pointer of a standard library function is unspecified. Which means extra caution should be taken before passing them as Callable. Instead it is typically better to wrap them in a lambda.
More on the topic: Can I take the address of a function defined in standard library?
However, C++20 introduced Constrained algorithms, or ranged algorithms, based on the Range-v3 library; where function-like entities, such as std::ranges::sort
and std::ranges::transform
, are introduced as Niebloids.
While the original library has created a functor class for each functions in the algorithm library, and each niebloids, such as ranges::sort
, is simply a named object of the corresponding functor class; the standard does not specify how they should be implemented.
So the question is if the behavior of passing a Niebloid as a Callable, such as std::invoke(std::ranges::sort, my_vec)
, specified/explicitly allowed?
All the spec says, in [algorithms.requirements] is:
The only way to implement that, today, is by making them objects. However, we don't specify any further behavior of those objects.
So this:
will work, simply because that will simply evaluate as
std::ranges::sort(my_vec)
after taking a reference to it, and there's no way to really prevent that from working.But other uses might not. For instance,
std::views::transform(r, std::ranges::distance)
is not specified to work, because we don't say whetherstd::ranges::distance
is copyable or not -std::ranges::size
is a customization point object, and thus copyable, butstd::ranges::distance
is just an algorithm.The MSVC implementation tries to adhere aggressively to the limited specification, and its implementation of
std::ranges::distance
is not copyable. libstdc++, on the other hand, just makes them empty objects, soviews::transform(ranges::distance)
just works by way of being not actively rejected.All of which to say is: once you get away from directly writing
std::ranges::meow(r)
(or otherwise writingmeow(r)
after ausing
orusing namespace
), you're kind of on your own.