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::result_of, std::invoke_result
std::result_of, std::invoke_result
class result_of; // ä¸å®ä¹
template< class F, class... ArgTypes >
(C++17 ä¸å¼ç¨)
(C++20 ä¸ç§»é¤)
class invoke_result;
å¨ç¼è¯æ¶æ¨å¯¼ INVOKE 表达å¼çè¿åç±»åã
|
|
(C++11 èµ·) (C++14 å) |
|
|
(C++14 èµ·) |
æ·»å æ¤é¡µé¢ä¸æè¿°ç任使¨¡æ¿çç¹åçç¨åºè¡ä¸ºæªå®ä¹ã
æåç±»å
| Â | |
| æåç±»å | å®ä¹ |
type
|
è¥ä»¥åæ° ArgTypes... è°ç¨å¯è°ç¨ (Callable) ç±»å F çè¿åç±»åãä»
è¥ F è½ä»¥åæ° ArgTypes... å¨ä¸æ±å¼è¯å¢ä¸è°ç¨æå¾å°å®ä¹ã (C++14 èµ·)
|
è¾ å©ç±»å
| template< class T > using result_of_t = typename result_of<T>::type; |
(1) | (C++14 èµ·) (C++17 ä¸å¼ç¨) (C++20 ä¸ç§»é¤) |
| template< class F, class... ArgTypes> using invoke_result_t = typename invoke_result<F, ArgTypes...>::type; |
(2) | (C++17 èµ·) |
å¯è½çå®ç°
namespace detail { template <class T> struct is_reference_wrapper : std::false_type {}; template <class U> struct is_reference_wrapper<std::reference_wrapper<U>> : std::true_type {};  template <class Base, class T, class Derived, class... Args> auto INVOKE(T Base::*pmf, Derived&& ref, Args&&... args) -> typename std::enable_if<std::is_function<T>::value && std::is_base_of<Base, typename std::decay<Derived>::type>::value, decltype((std::forward<Derived>(ref).*pmf)(std::forward<Args>(args)...))>::type;  template <class Base, class T, class RefWrap, class... Args> auto INVOKE(T Base::*pmf, RefWrap&& ref, Args&&... args) -> typename std::enable_if<std::is_function<T>::value && is_reference_wrapper<typename std::decay<RefWrap>::type>::value, decltype((ref.get().*pmf)(std::forward<Args>(args)...))>::type;  template <class Base, class T, class Pointer, class... Args> auto INVOKE(T Base::*pmf, Pointer&& ptr, Args&&... args) -> typename std::enable_if<std::is_function<T>::value && !is_reference_wrapper<typename std::decay<Pointer>::type>::value && !std::is_base_of<Base, typename std::decay<Pointer>::type>::value, decltype(((*std::forward<Pointer>(ptr)).*pmf)(std::forward<Args>(args)...))>::type;  template <class Base, class T, class Derived> auto INVOKE(T Base::*pmd, Derived&& ref) -> typename std::enable_if<!std::is_function<T>::value && std::is_base_of<Base, typename std::decay<Derived>::type>::value, decltype(std::forward<Derived>(ref).*pmd)>::type;  template <class Base, class T, class RefWrap> auto INVOKE(T Base::*pmd, RefWrap&& ref) -> typename std::enable_if<!std::is_function<T>::value && is_reference_wrapper<typename std::decay<RefWrap>::type>::value, decltype(ref.get().*pmd)>::type;  template <class Base, class T, class Pointer> auto INVOKE(T Base::*pmd, Pointer&& ptr) -> typename std::enable_if<!std::is_function<T>::value && !is_reference_wrapper<typename std::decay<Pointer>::type>::value && !std::is_base_of<Base, typename std::decay<Pointer>::type>::value, decltype((*std::forward<Pointer>(ptr)).*pmd)>::type;  template <class F, class... Args> auto INVOKE(F&& f, Args&&... args) -> typename std::enable_if<!std::is_member_pointer<typename std::decay<F>::type>::value, decltype(std::forward<F>(f)(std::forward<Args>(args)...))>::type; } // namespace detail  // æå° C++11 å®ç°ï¼ template <class> struct result_of; template <class F, class... ArgTypes> struct result_of<F(ArgTypes...)> { using type = decltype(detail::INVOKE(std::declval<F>(), std::declval<ArgTypes>()...)); };  // 符å C++14 çå®ç°ï¼äº¦ä¸ºåæ³ç C++11 å®ç°ï¼ï¼ namespace detail { template <typename AlwaysVoid, typename, typename...> struct invoke_result { }; template <typename F, typename...Args> struct invoke_result<decltype(void(detail::INVOKE(std::declval<F>(), std::declval<Args>()...))), F, Args...> { using type = decltype(detail::INVOKE(std::declval<F>(), std::declval<Args>()...)); }; } // namespace detail  template <class> struct result_of; template <class F, class... ArgTypes> struct result_of<F(ArgTypes...)> : detail::invoke_result<void, F, ArgTypes...> {};  template <class F, class... ArgTypes> struct invoke_result : detail::invoke_result<void, F, ArgTypes...> {};
注解
å¨ C++11 ä¸è§èæ¶ï¼ std::result_of çè¡ä¸ºå¨ INVOKE(std::declval<F>(), std::declval<ArgTypes>()...) 为ç
弿¶ï¼ä¾å¦ F å®å
¨ä¸å¯è°ç¨æ¶ï¼æªå®ä¹ã C++14 æ´æ¹ä¸º SFINAE ï¼ F ä¸å¯è°ç¨æ¶ï¼ std::result_of<F(ArgTypes...)> ç®åå°æ type æåï¼ã
std::result_of åç卿ºæ¯ç¡®å®è°ç¨å¯è°ç¨ (Callable) ç±»åçç»æï¼å°¤å
¶æ¯ç»æç±»å对ä¸ååæ°éä¸åçæ
åµã
F(Args...) æ¯ä»¥ Args... ä¸ºåæ°è以 F 为è¿åç±»åç彿°ç±»åãå èï¼ std::result_of æ¿åäºè®¸å¤æªå¼ï¼å¯¼è´å®å¨ C++17 ä¸è¢« std::invoke_result å代è被å¼ç¨ï¼
-
Fä¸è½æ¯å½æ°ç±»åææ°ç»ç±»åï¼ä½è½æ¯å°å®ä»¬çå¼ç¨ï¼ï¼ - è¥ä»»ä½
Argsæ¥æâTçæ°ç»âç±»åæå½æ°ç±»åTï¼åå®è¢«èªå¨è°æ´ä¸ºT*ï¼ -
Fæä»»ä½Args...é½ä¸è½æ¯æ½è±¡ç±»ç±»åï¼ - è¥ä»»ä½
Args...æ¥æé¡¶å± cv éå®ç¬¦ï¼åèå¼ä¹ï¼ -
Args...åä¸å¯ä¸º void ã
为é¿å
è¿äºæªå¼ï¼å¸¸ä»¥ F å Args... çå¼ç¨ç±»åä½¿ç¨ result_of ãä¾å¦ï¼
template<class F, class... Args> std::result_of_t<F&&(Args&&...)> // æ¿ä»£é误ç std::result_of_t<F(Args...)> my_invoke(F&& f, Args&&... args) { /* implementation */ }
示ä¾
#include <type_traits> #include <iostream>  struct S { double operator()(char, int&); float operator()(int) { return 1.0;} };  template<class T> typename std::result_of<T(int)>::type f(T& t) { std::cout << "overload of f for callable T\n"; return t(0); }  template<class T, class U> int f(U u) { std::cout << "overload of f for non-callable T\n"; return u; }  int main() { // 以 char å int åæ°è°ç¨ S çç»ææ¯ double std::result_of<S(char, int&)>::type d = 3.14; // d æ¥æ double ç±»å static_assert(std::is_same<decltype(d), double>::value, "");  // 以 int åæ°è°ç¨ S çç»ææ¯ float std::result_of<S(int)>::type x = 3.14; // x æ¥æ float ç±»å static_assert(std::is_same<decltype(x), float>::value, "");  // result_of è½ä»¥æåæå彿°çæé以å¦ä¸æ¹å¼ä½¿ç¨ struct C { double Func(char, int&); }; std::result_of<decltype(&C::Func)(C, char, int&)>::type g = 3.14; static_assert(std::is_same<decltype(g), double>::value, "");  f<C>(1); // C++11 ä¸å¯è½ç¼è¯å¤±è´¥ï¼ C++14 ä¸è°ç¨ä¸å¯è°ç¨éè½½ }
è¾åºï¼
overload of f for non-callable T
åé
| (C++17) |
以ç»å®å®åè°ç¨ä»»æå¯è°ç¨ (Callable) 对象 (彿°æ¨¡æ¿) |
| æ£æ¥ç±»åè½å¦ä»¥ç»å®çå®åç±»åè°ç¨ï¼å¦å以 std::invokeï¼ (类模æ¿) | |
| (C++11) |
è·åå°å
¶å®åçå¼ç¨ï¼ç¨äºä¸æ±å¼è¯å¢ä¸ (彿°æ¨¡æ¿) |