C++ 参考手册
- C++11
- C++14
- C++17
- C++20
- C++ 编译器支持情况表
- 独立与宿主实现
- C++ 语言
- C++ 关键词
- 预处理器
- C++ 标准库头文件
- 具名要求
- 功能特性测试 (C++20)
- 工具库
- 类型支持(基本类型、RTTI、类型特性)
- 概念库 (C++20)
- 错误处理
- 动态内存管理
- 日期和时间工具
- 字符串库
- 容器库
- 迭代器库
- 范围库 (C++20)
- 算法库
- 数值库
- 输入/输出库
- 文件系统库
- 本地化库
- 正则表达式库
- 原子操作库
- 线程支持库
- 实验性 C++ 特性
- 并行扩展
- 并行扩展,版本 2
- std::experimental::sample
- std::experimental::to_array
- std::experimental::source_location
- std::experimental::not_fn
- std::experimental::future
- std::experimental::shared_future
- C++ 标准库扩展
- std::experimental::pmr::resource_adaptor
- std::experimental::invocation_type, std::experimental::raw_invocation_type
- C++ 标准库扩展,版本 2
- std::experimental::propagate_const
- std::experimental::ostream_joiner
- std::experimental::randint
- std::experimental::observer_ptr
- std::experimental::is_detected, std::experimental::detected_t, std::experimental::detected_or
- C++ 标准库扩展,版本 3
- std::experimental::scope_exit
- std::experimental::scope_fail
- std::experimental::scope_success
- std::experimental::unique_resource
- 并发扩展
- 制约与概念
- 范围扩展
- std::experimental::function
- std::experimental::nonesuch
- std::experimental::reseed
- std::experimental::shuffle
- std::experimental::when_all
- std::experimental::barrier
- std::experimental::latch
- std::experimental::make_array
- 数学特殊函数
- 文件系统库
- std::experimental::erased_type
- std::experimental::when_any
- std::experimental::make_ready_future
- std::experimental::make_exceptional_future
- std::experimental::flex_barrier
- 注释
- 有用的资源
- 索引
- std 符号索引
- 协程支持 (C++20)
- C++ 关键词
位置:首页 > C++ 参考手册 >实验性 C++ 特性 > std::experimental::is_detected, std::experimental::detected_t, std::experimental::detected_or
std::experimental::is_detected, std::experimental::detected_t, std::experimental::detected_or
定义于头文件 <experimental/type_traits>
|
||
template< template<class...> class Op, class... Args > using is_detected = /* see below */; |
(库基础 TS v2) | |
template< template<class...> class Op, class... Args > using detected_t = /* see below */; |
(库基础 TS v2) | |
template< class Default, template<class...> class Op, class... Args > using detected_or = /* see below */; |
(库基础 TS v2) | |
别名模板 detected_or
是拥有二个公开成员 typedef value_t
和 type
的未指定类类型的别名,它们定义如下:
- 若模板 id Op<Args...> 代表合法类型,则
value_t
是 std::true_type 的别名,而type
是 Op<Args...> 的别名; - 否则,
value_t
是 std::false_type 的别名而type
是Default
的别名。
别名模板 is_detected
等价于 typename detected_or<std::experimental::nonesuch, Op, Args...>::value_t 。若模板 id Op<Args...> 代表合法类型,则它是 std::true_type 的别名;否则它是 std::false_type 的别名。
别名模版 detected_t
等价于 typename detected_or<std::experimental::nonesuch, Op, Args...>::type 。若模板 id Op<Args...> 代表合法类型,则它是 Op<Args...> 的别名;否则它是类 std::experimental::nonesuch 的别名。
附加工具
template< template<class...> class Op, class... Args > constexpr bool is_detected_v = is_detected<Op, Args...>::value; |
(库基础 TS v2) | |
template< class Default, template<class...> class Op, class... Args > using detected_or_t = typename detected_or<Default, Op, Args...>::type; |
(库基础 TS v2) | |
template <class Expected, template<class...> class Op, class... Args> using is_detected_exact = is_same<Expected, detected_t<Op, Args...>>; |
(库基础 TS v2) | |
template <class Expected, template<class...> class Op, class... Args> constexpr bool is_detected_exact_v = |
(库基础 TS v2) | |
template <class To, template<class...> class Op, class... Args> using is_detected_convertible = is_convertible<detected_t<Op, Args...>, To>; |
(库基础 TS v2) | |
template <class To, template<class...> class Op, class... Args> constexpr bool is_detected_convertible_v = |
(库基础 TS v2) | |
别名模版 is_detected_exact
检查 detected_t<Op, Args...> 是否为 Expected
。
别名模板 is_detected_convertible
检查 detected_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...> { // 注意 std::void_t 是 C++17 特性 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 <experimental/type_traits> #include <cstddef> 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() {}