命名空间
变体
操作

std::not_fn

来自 cppreference.com
< cpp‎ | utility‎ | functional
 
 
实用程序库
语言支持
类型支持 (基本类型,RTTI)
库特性测试宏 (C++20)
动态内存管理
程序实用程序
协程支持 (C++20)
可变参数函数
调试支持
(C++26)
三元比较
(C++20)
(C++20)(C++20)(C++20)
(C++20)(C++20)(C++20)
通用实用程序
日期和时间
函数对象
格式化库 (C++20)
(C++11)
关系运算符 (C++20 中已弃用)
整数比较函数
(C++20)(C++20)(C++20)   
(C++20)
交换类型操作
(C++14)
(C++11)
(C++11)
(C++11)
(C++17)
通用词汇类型
(C++11)
(C++17)
(C++17)
(C++17)
(C++11)
(C++17)
(C++23)
基本字符串转换
(C++17)
(C++17)

 
函数对象
函数调用
(C++17)(C++23)
标识函数对象
(C++20)
透明运算符包装器
(C++14)
(C++14)
(C++14)
(C++14)  
(C++14)
(C++14)
(C++14)
(C++14)
(C++14)
(C++14)
(C++14)
(C++14)
(C++14)

旧的绑定器和适配器
(直到 C++17*)
(直到 C++17*)
(直到 C++17*)
(直到 C++17*)  
(直到 C++17*)
(直到 C++17*)(直到 C++17*)(直到 C++17*)(直到 C++17*)
(直到 C++20*)
(直到 C++20*)
(直到 C++17*)(直到 C++17*)
(直到 C++17*)(直到 C++17*)

(直到 C++17*)
(直到 C++17*)(直到 C++17*)(直到 C++17*)(直到 C++17*)
(直到 C++20*)
(直到 C++20*)
 
定义在头文件 <functional>
template< class F >
/* 未指定 */ not_fn( F&& f );
(1) (自 C++17 起)
(自 C++20 起为 constexpr)
template< auto ConstFn >
constexpr /* 未指定 */ not_fn() noexcept;
(2) (自 C++26 起)
1) 创建一个转发调用包装器,它返回其持有的可调用对象的否定。
2) 创建一个转发调用包装器,它返回静态确定的可调用目标的否定。如果 ConstFn 是空指针或空成员指针,则程序格式不正确。

内容

[编辑] 参数

f - 包装器持有的 可调用 对象构建的 对象
类型要求
-
std::decay_t<F> 必须满足 可调用可移动构造 的要求。
-
std::is_constructible_v<std::decay_t<F>, F> 必须为 true

[编辑] 返回值

1) 一个类型为 T(未指定)的函数对象。它具有以下成员。

std::not_fn 返回类型

成员对象

std::not_fn 的返回类型包含一个类型为 std::decay_t<F> 的成员对象。

构造函数

explicit T( F&& f );
(1) (自 C++17 起)
(自 C++20 起为 constexpr)
(仅供说明*)
T( T&& f ) = default;
T( const T& f ) = default;
(2) (自 C++17 起)
1) 构造函数使用直接非列表初始化方式,将成员对象(类型为 std::decay_t<F>) 从 std::forward<F>(f) 初始化。抛出构造函数选择过程中抛出的任何异常。
2) 由于 std::decay_t<F> 必须是 可移动构造,因此返回的调用包装器始终是 可移动构造,如果 std::decay_t<F>可复制构造,则也是 可复制构造

显式默认定义使返回类型不可赋值。

(直到 C++20)

这些构造函数是否显式默认以及返回类型是否可赋值是未指定的。

(自 C++20)

成员函数 operator()

(1)
template< class... Args >

auto operator()( Args&&... args ) &
-> decltype(!std::declval<
    std::invoke_result_t<std::decay_t<F>&, Args...>>());
template< class... Args >
auto operator()( Args&&... args ) const&
-> decltype(!std::declval<

    std::invoke_result_t<std::decay_t<F> const&, Args...>>());
(自 C++17 起)
(直到 C++20)
template< class... Args >

constexpr auto operator()( Args&&... args ) &
    noexcept(/* 见下文 */)
-> decltype(!std::invoke(
    std::declval<std::decay_t<F>&>(), std::declval<Args>()...));
template< class... Args >
constexpr auto operator()( Args&&... args ) const&
    noexcept(/* 见下文 */)
-> decltype(!std::invoke(

    std::declval<std::decay_t<F> const&>(), std::declval<Args>()...));
(自 C++20)
(2)
template< class... Args >

auto operator()( Args&&... args ) &&
-> decltype(!std::declval<
    std::invoke_result_t<std::decay_t<F>, Args...>>());
template< class... Args >
auto operator()( Args&&... args ) const&&
-> decltype(!std::declval<

    std::invoke_result_t<std::decay_t<F> const, Args...>>());
(自 C++17 起)
(直到 C++20)
template< class... Args >

constexpr auto operator()( Args&&... args ) &&
    noexcept(/* 见下文 */)
-> decltype(!std::invoke(
    std::declval<std::decay_t<F>>(), std::declval<Args>()...));
template< class... Args >
constexpr auto operator()( Args&&... args ) const&&
    noexcept(/* 见下文 */)
-> decltype(!std::invoke(

    std::declval<std::decay_t<F> const>(), std::declval<Args>()...));
(自 C++20)

fd 为类型为 std::decay_t<F> 的成员对象。

1) 等同于 return !std::invoke(fd, std::forward<Args>(args)...);
2) 等同于 return !std::invoke(std::move(fd), std::forward<Args>(args)...);

在调用结果时,如果将原始选定的 operator() 重载的返回值类型进行替换失败,可以选择另一个重载

(自 C++17 起)
(直到 C++20)
2) 表达式等同于 !std::invoke(std::move(fd), std::forward<Args>(args)...)

在调用结果时,如果将原始选定的 operator() 重载的返回值类型进行替换失败,则调用格式错误,这也可能是一个 替换失败

(自 C++20)
2) 以下类型的值。

std::not_fn 无状态返回类型

返回类型是一个 可复制构造 无状态类。返回类型是否可赋值是未指定的。

成员函数 operator()

template< class... Args >

constexpr auto operator()( Args&&... args ) const
    noexcept(/* 见下文 */)

-> decltype(!std::invoke(ConstFn, std::declval<Args>()...));
(自 C++26 起)

表达式等同于 !std::invoke(ConstFn, std::forward<Args>(args)...).

[编辑] 异常

1) 不抛出异常,除非 fd 的构造抛出异常。

[编辑] 可能的实现

(1) not_fn
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 since C++20
        // 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)};
}
(2) not_fn
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 {};
}

[编辑] 说明

std::not_fn 用于替换 C++03 时代的否定器 std::not1std::not2

功能测试 标准 功能
__cpp_lib_not_fn 201603L (C++17) std::not_fn(), (1)
202306L (C++26) 允许将可调用对象作为非类型模板参数传递给 std::not_fn, (2)

[编辑] 示例

#include <cassert>
#include <functional>
 
bool is_same(int a, int b) noexcept
{
    return a == b;
}
 
struct S
{
    int val;
    bool is_same(int arg) const noexcept { return val == arg; }
};
 
int main()
{
    // Using with a free function:
    auto is_differ = std::not_fn(is_same);
    assert(is_differ(8, 8) == false); // equivalent to: !is_same(8, 8) == false
    assert(is_differ(6, 9) == true); // equivalent to: !is_same(8, 0) == true
 
    // Using with a member function:
    auto member_differ = std::not_fn(&S::is_same);
    assert(member_differ(S{3}, 3) == false); //: S tmp{6}; !tmp.is_same(6) == false
 
    // Noexcept-specification is preserved:
    static_assert(noexcept(is_differ) == noexcept(is_same));
    static_assert(noexcept(member_differ) == noexcept(&S::is_same));
 
    // Using with a function object:
    auto same = [](int a, int b) { return a == b; };
    auto differ = std::not_fn(same);
    assert(differ(1, 2) == true); //: !same(1, 2) == true
    assert(differ(2, 2) == false); //: !same(2, 2) == false
 
#if __cpp_lib_not_fn >= 202306L
    auto is_differ_cpp26 = std::not_fn<is_same>();
    assert(is_differ_cpp26(8, 8) == false);
    assert(is_differ_cpp26(6, 9) == true);
 
    auto member_differ_cpp26 = std::not_fn<&S::is_same>();
    assert(member_differ_cpp26(S{3}, 3) == false);
 
    auto differ_cpp26 = std::not_fn<same>();
    static_assert(differ_cpp26(1, 2) == true);
    static_assert(differ_cpp26(2, 2) == false);
#endif
}

[编辑] 参见

(C++17 中已弃用)(C++20 中已移除)
构造自定义的 std::unary_negate 对象
(函数模板) [编辑]
(C++17 中已弃用)(C++20 中已移除)
构造自定义的 std::binary_negate 对象
(函数模板) [编辑]