std::not_fn
Defined in header <functional>
|
||
template< class F > /* unspecified */ not_fn( F&& f ); |
(1) | (since C++17) (constexpr since C++20) |
template< auto ConstFn > constexpr /* unspecified */ not_fn() noexcept; |
(2) | (since C++26) |
ConstFn
is a null pointer or null pointer-to-member.Contents |
Parameters
f | - | the object from which the Callable object held by the wrapper is constructed |
Type requirements | ||
-std::decay_t<F> must meet the requirements of Callable and MoveConstructible.
| ||
-std::is_constructible_v<std::decay_t<F>, F> is required to be true. |
Return value
A function object of unspecified type T
. It has the following members:
std::not_fn return type
Member objects
The return type of std::not_fn
holds a member object of type std::decay_t<F>.
Constructors
explicit T( F&& f ); |
(1) | (since C++17) (constexpr since C++20) (exposition only*) |
T( T&& f ) = default; T( const T& f ) = default; |
(2) | |
It is unspecified whether the return type is assignable.
Member function operator()
(1) | ||
template< class... Args > auto operator()( Args&&... args ) & |
(since C++17) (until C++20) |
|
template< class... Args > constexpr auto operator()( Args&&... args ) & |
(since C++20) | |
(2) | ||
template< class... Args > auto operator()( Args&&... args ) && |
(since C++17) (until C++20) |
|
template< class... Args > constexpr auto operator()( Args&&... args ) && |
(since C++20) | |
1) Equivalent to return !std::invoke(fd, std::forward<Args>(args)...);
2) Equivalent to return !std::invoke(std::move(fd), std::forward<Args>(args)...);
|
(since C++17) (until C++20) |
(since C++20) |
where fd is the member object of type std::decay_t<F>.
std::not_fn stateless return type
The return type is a CopyConstructible stateless class. It is unspecified whether the return type is assignable.
Member function operator()
template< class... Args > constexpr auto operator()( Args&&... args ) const |
(since C++26) | |
Expression-equivalent to !std::invoke(ConstFn, std::forward<Args>(args)...).
Exceptions
Throws no exceptions, unless the construction of fd throws.
Possible implementation
First version |
---|
namespace detail { template<class V, class F, class... Args> constexpr bool negate_invocable_impl = false; template<class F, class... Args> constexpr bool negate_invocable_impl<std::void_t<decltype( !std::invoke(std::declval<F>(), std::declval<Args>()...))>, F, Args...> = true; template<class F, class... Args> constexpr bool negate_invocable_v = negate_invocable_impl<void, F, Args...>; template<class F> struct not_fn_t { F f; template<class... Args, std::enable_if_t<negate_invocable_v<F&, Args...>, int> = 0> constexpr decltype(auto) operator()(Args&&... args) & noexcept(noexcept(!std::invoke(f, std::forward<Args>(args)...))) { return !std::invoke(f, std::forward<Args>(args)...); } template<class... Args, std::enable_if_t<negate_invocable_v<const F&, Args...>, int> = 0> constexpr decltype(auto) operator()(Args&&... args) const& noexcept(noexcept(!std::invoke(f, std::forward<Args>(args)...))) { return !std::invoke(f, std::forward<Args>(args)...); } template<class... Args, std::enable_if_t<negate_invocable_v<F, Args...>, int> = 0> constexpr decltype(auto) operator()(Args&&... args) && noexcept(noexcept(!std::invoke(std::move(f), std::forward<Args>(args)...))) { return !std::invoke(std::move(f), std::forward<Args>(args)...); } template<class... Args, std::enable_if_t<negate_invocable_v<const F, Args...>, int> = 0> constexpr decltype(auto) operator()(Args&&... args) const&& noexcept(noexcept(!std::invoke(std::move(f), std::forward<Args>(args)...))) { return !std::invoke(std::move(f), std::forward<Args>(args)...); } // Deleted overloads are needed for preventing a non-equivalent but well-formed // overload to be selected. template<class... Args, std::enable_if_t<!negate_invocable_v<F&, Args...>, int> = 0> void operator()(Args&&...) & = delete; template<class... Args, std::enable_if_t<!negate_invocable_v<const F&, Args...>, int> = 0> void operator()(Args&&...) const& = delete; template<class... Args, std::enable_if_t<!negate_invocable_v<F, Args...>, int> = 0> void operator()(Args&&...) && = delete; template<class... Args, std::enable_if_t<!negate_invocable_v<const F, Args...>, int> = 0> void operator()(Args&&...) const&& = delete; }; } template<class F> constexpr detail::not_fn_t<std::decay_t<F>> not_fn(F&& f) { return {std::forward<F>(f)}; } |
Second version |
namespace detail { template<auto ConstFn> struct stateless_not_fn { template<class... Args> constexpr auto operator()(Args&&... args) const noexcept(noexcept(!std::invoke(ConstFn, std::forward<Args>(args)...))) -> decltype(!std::invoke(ConstFn, std::forward<Args>(args)...)) { return !std::invoke(ConstFn, std::forward<Args>(args)...); } }; } template<auto ConstFn> constexpr detail::stateless_not_fn<ConstFn> not_fn() noexcept { if constexpr (std::is_pointer_v<decltype(ConstFn)> || std::is_member_pointer_v<decltype(ConstFn)>) { static_assert(ConstFn != nullptr); } return {}; } |
Notes
std::not_fn
is intended to replace the C++03-era negators std::not1 and std::not2.
Feature-test macro | Value | Std | Feature |
---|---|---|---|
__cpp_lib_not_fn |
201603L | (C++17) | std::not_fn()
|
202306L | (C++26) | Allow passing callable objects as non-type template arguments to std::not_fn
|
Example
This section is incomplete Reason: no example |
See also
(deprecated in C++17)(removed in C++20) |
constructs custom std::unary_negate object (function template) |
(deprecated in C++17)(removed in C++20) |
constructs custom std::binary_negate object (function template) |