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::common_type
std::common_type
| å®ä¹äºå¤´æä»¶ <type_traits>
|
||
| template< class... T > struct common_type; |
(C++11 èµ·) | |
ç¡®å®ææç±»å T... çå
±ç¨ç±»åï¼å³ææ T... é½è½éå¼è½¬æ¢å°çç±»åãè¥è¿ç§ç±»åï¼æ ¹æ®åè¿°è§åç¡®å®ï¼åå¨ï¼åæå type æå该类åãå¦åï¼æ æå type ã
- è¥ sizeof...(T) 为é¶ï¼åæ æå
typeã - è¥ sizeof...(T) 为ä¸ï¼å³
T...åªå«ä¸ä¸ªç±»åT0ï¼ï¼åæåtypeæåä¸ std::common_type<T0, T0>::type ç¸åçç±»åï¼è¥å®åå¨ï¼å¦åæ æåtypeã - è¥ sizeof...(T) 为äºï¼å³
T...嫿°å¥½äºä¸ªæåT1ä¸T2ï¼ï¼å
- è¥åºç¨ std::decay å°è³å°
T1ä¸T2ä¸è³å°ä¸ä¸ªç±»åå产çç¸å¼ç±»åï¼åæåtypeæåä¸ std::common_type<std::decay<T1>::type, std::decay<T2>::type>::type ç¸åçç±»åï¼è¥å®åå¨ï¼è¥ä¸åå¨ï¼åæ æåtypeã - å¦åï¼è¥æå¯¹ std::common_type<T1, T2> çç¨æ·å®ä¹ç¹åï¼å使ç¨è¯¥ç¹åï¼
- å¦åï¼ è¥ std::decay<decltype(false ? std::declval<T1>() : std::declval<T2>())>::type æ¯åæ³ç±»åï¼åæå
type代表该类åï¼
- è¥åºç¨ std::decay å°è³å°
|
(C++20 èµ·) |
- å¦åï¼æ æå
typeã
- å¦åï¼æ æå
- è¥ sizeof...(T) 大äºäºï¼å³
T...ç±ç±»åT1, T2, R...ç»æï¼ï¼åè¥ std::common_type<T1, T2>::type åå¨ï¼åæåtypeæä»£ std::common_type<std::common_type<T1, T2>::type, R...>::type ï¼è¥åå¨è¿ç§ç±»åãå ¶ä»æææ åµä¸ï¼æ æåtypeã
åæ°å
T ä¸çç±»ååºåä¸ºå®æ´ç±»åãï¼å¯ä¸º cv éå®çï¼ void ï¼ææªç¥è¾¹çæ°ç»ãå¦åè¡ä¸ºæªå®ä¹ã
è¥ä¸è¿°æ¨¡æ¿çå®ä¾åç´æ¥æé´æ¥å°ä¾èµäºä¸å®æ´ç±»åï¼å¹¶ä¸å¦æåå¦ä½¿è¯¥ç±»å宿´ï¼å®ä¾åå°±ä¼äº§çä¸åçç»æï¼åè¡ä¸ºæªå®ä¹ã
æåç±»å
| Â | |
| åç§° | å®ä¹ |
type
|
ææ T... çå
±ç¨ç±»å
|
è¾ å©ç±»å
| template< class... T > using common_type_t = typename common_type<T...>::type; |
(C++14 èµ·) | |
ç¹å
ç¨æ·å¯ä¸ºç±»å T1 ä¸ T2 ç¹å common_type ï¼åªè¦
-
T1ä¸T2è³å°ä¸ä¸ªä¾èµäºç¨æ·å®ä¹ç±»åï¼ä¸ - std::decay 对äº
T1åT2齿¯åç忢ã
è¥è¿ç§ç¹åæ¥æå为 type çæåï¼åå®å¿
é¡»æ¯å
¬å¼ä¸æ æ§ä¹çæåç±»åï¼å¹¶æåä¸ä¸ª T1 å T2 é½è½æ¾å¼è½¬æ¢å°çï¼æ cv éå®ä¸éå¼ç¨çç±»åãèä¸ï¼ std::common_type<T1, T2>::type ä¸ std::common_type<T2, T1>::type å¿
é¡»æä»£åä¸ç±»åã
以è¿èè¿äºè§åçæ¹å¼æ·»å common_type ç¹åçç¨åºæ¥ææªå®ä¹è¡ä¸ºã
æ³¨ææ·»å ç¹åå°æ¥èª <type_traits> çä»»ä½å
¶ä»æ¨¡æ¿çè¡ä¸ºæ¯æªå®ä¹çã
æ ååºå·²æä¾ä¸åç¹åï¼
| ç¹å std::common_type ç¹å¾ (类模æ¿ç¹å) | |
| ç¹å std::common_type ç¹å¾ (类模æ¿ç¹å) |
å¯è½çå®ç°
template <typename...> using void_t = void;  // åçæ¨¡æ¿ï¼ç¨äºé¶ä¸ªç±»åï¼ template <class...> struct common_type {};  //////// ä¸ä¸ªç±»å template <class T> struct common_type<T> : common_type<T, T> {};  //////// äºä¸ªç±»å template <class T1, class T2> using cond_t = decltype(false ? std::declval<T1>() : std::declval<T2>());  template <class T1, class T2, class=void> struct common_type_2_impl {};  template <class T1, class T2> struct common_type_2_impl<T1, T2, void_t<cond_t<T1, T2>>> { using type = typename std::decay<cond_t<T1, T2>>::type; };  template <class T1, class T2> struct common_type<T1, T2> : common_type_2_impl<typename std::decay<T1>::type, typename std::decay<T2>::type> {};  //////// 3 个以ä¸ç±»å template <class AlwaysVoid, class T1, class T2, class...R> struct common_type_multi_impl {};  template <class T1, class T2, class...R> struct common_type_multi_impl< void_t<typename common_type<T1, T2>::type>, T1, T2, R...> : common_type<typename common_type<T1, T2>::type, R...> {};   template <class T1, class T2, class... R> struct common_type<T1, T2, R...> : common_type_multi_impl<void, T1, T2, R...> {}; |
注æ
对äºä¸æ¯æåç®æ çç®æ¯ç±»åï¼å ±ç¨ç±»åå¯è§ä¸ºå¦ T0() + T1() + ... + Tn() çï¼å¯ä»¥ä¸ºæ··å模å¼çï¼ç®æ¯è¡¨è¾¾å¼çç±»åã
ç¼ºé·æ¥å
ä¸åæ´æ¹è¡ä¸ºçç¼ºé·æ¥å追溯å°åºç¨äºä»¥ååºçç C++ æ åã
| DR | åºç¨äº | åºçæ¶çè¡ä¸º | æ£ç¡®è¡ä¸º |
|---|---|---|---|
| LWG 2141 | C++11 | common_type<int, int>::type 为 int&& | éåäºç»æç±»å |
| LWG 2408 | C++11 | common_type é SFINAE å好
|
令ä¹ä¸º SFINAE å好 |
| LWG 2460 | C++11 | common_type ç¹åç书åè¿ä¹ä¸å¯è½
|
åå°äºéè¦çç¹åæ°é |
示ä¾
æ¼ç¤ºç¨æ·å®ä¹ç±»ä¸çæ··åæ¨¡å¼ç®æ¯
#include <iostream> #include <type_traits> Â template <class T> struct Number { T n; }; Â template <class T, class U> Number<typename std::common_type<T, U>::type> operator+(const Number<T>& lhs, const Number<U>& rhs) { return {lhs.n + rhs.n}; } Â int main() { Number<int> i1 = {1}, i2 = {2}; Number<double> d1 = {2.3}, d2 = {3.5}; std::cout << "i1i2: " << (i1 + i2).n << "\ni1d2: " << (i1 + d2).n << '\n' << "d1i2: " << (d1 + i2).n << "\nd1d2: " << (d1 + d2).n << '\n'; }
è¾åºï¼
i1i2: 3 i1d2: 4.5 d1i2: 4.3 d1d2: 5.8