C++ åèæå
- C++11
- C++14
- C++17
- C++20
- C++ ç¼è¯å¨æ¯ææ åµè¡¨
- ç¬ç«ä¸å®¿ä¸»å®ç°
- C++ è¯è¨
- C++ å ³é®è¯
- é¢å¤çå¨
- C++ æ ååºå¤´æä»¶
- å ·åè¦æ±
- åè½ç¹æ§æµè¯ (C++20)
- å·¥å ·åº
- ç±»åæ¯æï¼åºæ¬ç±»åãRTTIãç±»åç¹æ§ï¼
- std::result_of, std::invoke_result
- std::integral_constant
- std::is_constant_evaluated
- std::max_align_t
- offsetof
- NULL
- std::bad_typeid
- std::bad_cast
- std::numeric_limits
- std::type_info
- std::ptrdiff_t
- std::byte
- std::conjunction
- std::disjunction
- std::negation
- std::is_swappable_with, std::is_swappable, std::is_nothrow_swappable_with, std::is_nothrow_swappable
- std::is_invocable, std::is_invocable_r, std::is_nothrow_invocable, std::is_nothrow_invocable_r
- std::is_aggregate
- std::has_unique_object_representations
- std::endian
- std::remove_cvref
- std::type_index
- C æ°å¼æéæ¥å£
- å®å®½æ´æ°ç±»å (C++11 èµ·)
- std::is_bounded_array
- std::is_unbounded_array
- std::size_t
- std::nullptr_t
- std::is_integral
- std::rank
- std::is_void
- std::is_null_pointer
- std::is_array
- std::is_pointer
- std::is_enum
- std::is_union
- std::is_class
- std::is_function
- std::is_object
- std::is_scalar
- std::is_compound
- std::is_floating_point
- std::is_fundamental
- std::is_arithmetic
- std::is_reference
- std::is_lvalue_reference
- std::is_rvalue_reference
- std::is_member_pointer
- std::is_member_object_pointer
- std::is_member_function_pointer
- std::is_const
- std::is_volatile
- std::is_empty
- std::is_polymorphic
- std::is_final
- std::is_abstract
- std::is_trivial
- std::is_trivially_copyable
- std::is_standard_layout
- std::is_literal_type
- std::is_pod
- std::is_signed
- std::is_unsigned
- std::is_constructible, std::is_trivially_constructible, std::is_nothrow_constructible
- std::is_default_constructible, std::is_trivially_default_constructible, std::is_nothrow_default_constructible
- std::is_copy_constructible, std::is_trivially_copy_constructible, std::is_nothrow_copy_constructible
- std::is_move_constructible, std::is_trivially_move_constructible, std::is_nothrow_move_constructible
- std::is_assignable, std::is_trivially_assignable, std::is_nothrow_assignable
- std::is_copy_assignable, std::is_trivially_copy_assignable, std::is_nothrow_copy_assignable
- std::is_move_assignable, std::is_trivially_move_assignable, std::is_nothrow_move_assignable
- std::is_destructible, std::is_trivially_destructible, std::is_nothrow_destructible
- std::has_virtual_destructor
- std::is_same
- std::is_base_of
- std::is_convertible, std::is_nothrow_convertible
- std::is_layout_compatible
- std::is_pointer_interconvertible_base_of
- std::is_pointer_interconvertible_with_class
- std::is_corresponding_member
- std::alignment_of
- std::extent
- std::remove_cv, std::remove_const, std::remove_volatile
- std::add_cv, std::add_const, std::add_volatile
- std::make_signed
- std::make_unsigned
- std::remove_reference
- std::add_lvalue_reference, std::add_rvalue_reference
- std::remove_pointer
- std::add_pointer
- std::remove_extent
- std::remove_all_extents
- std::aligned_storage
- std::aligned_union
- std::decay
- std::enable_if
- std::void_t
- std::conditional
- std::common_type
- std::common_reference
- std::underlying_type
- std::type_identity
- 注é
- æ¦å¿µåº (C++20)
- é误å¤ç
- 卿å å管ç
- æ¥æåæ¶é´å·¥å ·
- å符串åº
- 容å¨åº
- è¿ä»£å¨åº
- èå´åº (C++20)
- ç®æ³åº
- æ°å¼åº
- è¾å ¥/è¾åºåº
- æä»¶ç³»ç»åº
- æ¬å°ååº
- æ£å表达å¼åº
- ååæä½åº
- çº¿ç¨æ¯æåº
- å®éªæ§ C++ ç¹æ§
- æç¨çèµæº
- ç´¢å¼
- std 符å·ç´¢å¼
- åç¨æ¯æ (C++20)
- C++ å ³é®è¯
ä½ç½®ï¼é¦é¡µ > C++ åèæå >ç±»åæ¯æï¼åºæ¬ç±»åãRTTIãç±»åç¹æ§ï¼ > std::void_t
std::void_t
| å®ä¹äºå¤´æä»¶ <type_traits>
|
||
| template< class... > using void_t = void; |
(C++17 èµ·) | |
å°ä»»æç±»åçåºåæ å°å°ç±»å void çå·¥å ·å 彿°ã
注æ
模æ¿å ç¼ç¨ä¸ï¼ç¨æ¤å 彿°æ£æµ SFINAE è¯å¢ä¸çç å¼ç±»åï¼
// åçæ¨¡æ¿å¤çæ åµå¥Â ::type æåçç±»åï¼ template< class, class = std::void_t<> > struct has_type_member : std::false_type { };  // ç¹åè¯å«æ¥æåµå¥Â ::type æåçç±»åï¼ template< class T > struct has_type_member<T, std::void_t<typename T::type>> : std::true_type { };
å®äº¦å¯ç¨äºæ£æµè¡¨è¾¾å¼çåæ³æ§ï¼
// åçæ¨¡æ¿å¤ç䏿¯æåç½®èªå¢çç±»åï¼ template< class, class = std::void_t<> > struct has_pre_increment_member : std::false_type { }; // ç¹åè¯å«æ¯æåç½®èªå¢çç±»åï¼ template< class T > struct has_pre_increment_member<T, std::void_t<decltype( ++std::declval<T&>() )> > : std::true_type { };
ç´è³ CWG1558 ï¼ C++14 缺é·ï¼åï¼å«å模æ¿ä¸æªä½¿ç¨çåæ°ä¸ä¿è¯ SFINAE ä¸å¯è¢«å¿½ç¥ï¼æ
æ©æç¼è¯å¨è¦æ± void_t çæ´å¤æå®ä¹ï¼ä¾å¦ï¼
template<typename... Ts> struct make_void { typedef void type;}; template<typename... Ts> using void_t = typename make_void<Ts...>::type;
示ä¾
è¿è¡æ¤ä»£ç
#include <iostream> #include <type_traits> #include <vector> #include <map>  class A {};  template <typename T, typename = void> struct is_iterable : std::false_type {}; template <typename T> struct is_iterable<T, std::void_t<decltype(std::declval<T>().begin()), decltype(std::declval<T>().end())>> : std::true_type {};  // è¿ä»£å¨ç¹æ§ï¼å ¶ value_type å§ç»ä¸ºè¢«è¿ä»£å®¹å¨ç value_type ï¼ // å³ä½¿ç¨ value_type 为 void ç back_insert_iterator  template <typename T, typename = void> struct iterator_trait : std::iterator_traits<T> {}; template <typename T> struct iterator_trait<T, std::void_t<typename T::container_type>> : std::iterator_traits<typename T::container_type::iterator> {};  int main() { std::cout << std::boolalpha; std::cout << is_iterable<std::vector<double>>::value << '\n'; std::cout << is_iterable<std::map<int, double>>::value << '\n'; std::cout << is_iterable<double>::value << '\n'; std::cout << is_iterable<A>::value << '\n';   std::vector<int> v;  std::cout << std::is_same<iterator_trait<decltype(std::back_inserter(v))>::value_type , iterator_trait<decltype(v.cbegin())>::value_type >::value << '\n'; }
è¾åºï¼
true true false false true
åé
| (C++11) |
æ ¹æ®ç¼è¯æå¸å°å¸¸ééèä¸ä¸ªå½æ°éè½½ææ¨¡æ¿ç¹å (类模æ¿) |