名前空間
変種
操作

std::experimental::is_detected, std::experimental::detected_t, std::experimental::detected_or

From cppreference.com
 
 
 
 
ヘッダ <experimental/type_traits> で定義
template< template<class...> class Op, class... Args >
using is_detected = /* 以下を参照 */;
(Library Fundamentals TS v2)
template< template<class...> class Op, class... Args >
using detected_t = /* 以下を参照 */;
(Library Fundamentals TS v2)
template< class Default, template<class...> class Op, class... Args >
using detected_or = /* 以下を参照 */;
(Library Fundamentals TS v2)

エイリアステンプレート detected_or は、2つの公開メンバ typedef value_ttype を持つ未規定のクラス型へのエイリアスであり、それらは以下のように定義される。

  • もしテンプレートID Op<Args...> が有効な型を表す場合、value_tstd::true_type へのエイリアスとなり、typeOp<Args...> へのエイリアスとなる。
  • そうでなければ、value_tstd::false_type へのエイリアスとなり、typeDefault へのエイリアスとなる。

エイリアステンプレート is_detectedtypename detected_or<std::experimental::nonesuch, Op, Args...>::value_t と等価である。これは、テンプレートID Op<Args...> が有効な型を表す場合は std::true_type へのエイリアスとなり、そうでなければ std::false_type へのエイリアスとなる。

エイリアステンプレート detected_ttypename detected_or<std::experimental::nonesuch, Op, Args...>::type と等価である。これは、そのテンプレートIDが有効な型を表す場合は Op<Args...> へのエイリアスとなり、そうでなければクラス std::experimental::nonesuch へのエイリアスとなる。

[編集] 追加のユーティリティ

template< template<class...> class Op, class... Args >
constexpr bool is_detected_v = is_detected<Op, Args...>::value;
(Library Fundamentals TS v2)
template< template<class...> class Op, class... Args >
constexpr inline bool is_detected_v = is_detected<Op, Args...>::value;
(ライブラリ基本TS v3)
template< class Default, template<class...> class Op, class... Args >
using detected_or_t = typename detected_or<Default, Op, Args...>::type;
(Library Fundamentals TS v2)
template< class Expected, template<class...> class Op, class... Args >
using is_detected_exact = std::is_same<Expected, detected_t<Op, Args...>>;
(Library Fundamentals TS v2)
template< class Expected, template<class...> class Op, class... Args >

constexpr bool is_detected_exact_v =

    is_detected_exact<Expected, Op, Args...>::value;
(Library Fundamentals TS v2)
template< class Expected, template<class...> class Op, class... Args >

constexpr inline bool is_detected_exact_v =

    is_detected_exact<Expected, Op, Args...>::value;
(ライブラリ基本TS v3)
template< class To, template<class...> class Op, class... Args >

using is_detected_convertible =

    std::is_convertible<detected_t<Op, Args...>, To>;
(Library Fundamentals TS v2)
template< class To, template<class...> class Op, class... Args >

constexpr bool is_detected_convertible_v =

    is_detected_convertible<To, Op, Args...>::value;
(Library Fundamentals TS v2)
template< class To, template<class...> class Op, class... Args >

constexpr inline bool is_detected_convertible_v =

    is_detected_convertible<To, Op, Args...>::value;
(ライブラリ基本TS v3)

エイリアステンプレート is_detected_exactdetected_t<Op, Args...>Expected と正確に同じ型であるかをチェックする。

エイリアステンプレート is_detected_convertibledetected_t<Op, Args...>To へ変換可能であるかをチェックする。

[編集] 実装例

namespace detail
{
    template<class Default, class AlwaysVoid, template<class...> class Op, class... Args>
    struct detector
    {
        using value_t = std::false_type;
        using type = Default;
    };
 
    template<class Default, template<class...> class Op, class... Args>
    struct detector<Default, std::void_t<Op<Args...>>, Op, Args...>
    {
        using value_t = std::true_type;
        using type = Op<Args...>;
    };
} // namespace detail
 
template<template<class...> class Op, class... Args>
using is_detected = typename detail::detector<nonesuch, void, Op, Args...>::value_t;
 
template<template<class...> class Op, class... Args>
using detected_t = typename detail::detector<nonesuch, void, Op, Args...>::type;
 
template<class Default, template<class...> class Op, class... Args>
using detected_or = detail::detector<Default, void, Op, Args...>;

[編集]

#include <cstddef>
#include <experimental/type_traits>
 
template<class T>
using copy_assign_t = decltype(std::declval<T&>() = std::declval<const T&>());
 
struct Meow {};
struct Purr { void operator=(const Purr&) = delete; };
 
static_assert(std::experimental::is_detected<copy_assign_t, Meow>::value,
              "Meow should be copy assignable!");
static_assert(!std::experimental::is_detected_v<copy_assign_t, Purr>,
              "Purr should not be copy assignable!");
static_assert(std::experimental::is_detected_exact_v<Meow&, copy_assign_t, Meow>,
              "Copy assignment of Meow should return Meow&!");
 
template<class T>
using diff_t = typename T::difference_type;
 
template<class Ptr>
using difference_type = std::experimental::detected_or_t<std::ptrdiff_t, diff_t, Ptr>;
 
struct Woof { using difference_type = int; };
struct Bark {};
 
static_assert(std::is_same<difference_type<Woof>, int>::value,
              "Woof's difference_type should be int!");
static_assert(std::is_same<difference_type<Bark>, std::ptrdiff_t>::value,
              "Bark's difference_type should be ptrdiff_t!");
 
int main() {}
English 日本語 中文(简体) 中文(繁體)