名前空間
変種
操作

std::not_fn

From 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++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 - ラッパーが保持するCallableオブジェクトが構築される元となるオブジェクト
型要件
-
std::decay_t<F>は、CallableおよびMoveConstructibleの要件を満たす必要があります。
-
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>MoveConstructibleである必要があるため、返される呼び出しラッパーは常にMoveConstructibleであり、std::decay_t<F>CopyConstructibleであればCopyConstructibleです。

明示的にデフォルト化された定義により、戻り値の型は代入不可になります。

(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 ステートレスな戻り値の型

戻り値の型はCopyConstructibleなステートレスクラスです。戻り値の型が代入可能であるかは未指定です。

メンバ関数 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::not1およびstd::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 オブジェクトを構築する
(関数テンプレート) [編集]
English 日本語 中文(简体) 中文(繁體)