C++ åèæå
- C++11
- C++14
- C++17
- C++20
- C++ ç¼è¯å¨æ¯ææ åµè¡¨
- ç¬ç«ä¸å®¿ä¸»å®ç°
- C++ è¯è¨
- C++ å ³é®è¯
- é¢å¤çå¨
- C++ æ ååºå¤´æä»¶
- å ·åè¦æ±
- åè½ç¹æ§æµè¯ (C++20)
- å·¥å ·åº
- std::apply
- åºç¹æ§æµè¯å® (C++20)
- ç¨åºæ¯æå·¥å ·
- std::initializer_list
- 彿°å¯¹è±¡
- std::hash
- std::pair
- std::tuple
- std::optional
- std::any
- std::variant
- æ ¼å¼ååº (C++20)
- std::integer_sequence
- std::exchange
- std::make_from_tuple
- std::launder
- std::to_chars
- std::from_chars
- std::as_const
- std::source_location
- å忰彿°
- std::bitset
- std::cmp_equal, cmp_not_equal, cmp_less, cmp_greater, cmp_less_equal, cmp_greater_equal
- std::in_range
- std::declval
- std::forward
- std::move
- std::move_if_noexcept
- std::chars_format
- std::piecewise_construct_t
- std::piecewise_construct
- std::in_place, std::in_place_type, std::in_place_index, std::in_place_t, std::in_place_type_t, std::in_place_index_t
- 注é
- ç±»åæ¯æï¼åºæ¬ç±»åãRTTIãç±»åç¹æ§ï¼
- æ¦å¿µåº (C++20)
- é误å¤ç
- 卿å å管ç
- æ¥æåæ¶é´å·¥å ·
- å符串åº
- 容å¨åº
- è¿ä»£å¨åº
- èå´åº (C++20)
- ç®æ³åº
- æ°å¼åº
- è¾å ¥/è¾åºåº
- æä»¶ç³»ç»åº
- æ¬å°ååº
- æ£å表达å¼åº
- ååæä½åº
- çº¿ç¨æ¯æåº
- å®éªæ§ C++ ç¹æ§
- æç¨çèµæº
- ç´¢å¼
- std 符å·ç´¢å¼
- åç¨æ¯æ (C++20)
- C++ å ³é®è¯
ä½ç½®ï¼é¦é¡µ > C++ åèæå >å·¥å ·åº > std::forward
std::forward
| å®ä¹äºå¤´æä»¶ <utility>
|
||
| (1) | ||
| template< class T > T&& forward( typename std::remove_reference<T>::type& t ) noexcept; |
(C++11 èµ·) (C++14 å) |
|
| template< class T > constexpr T&& forward( std::remove_reference_t<T>& t ) noexcept; |
(C++14 èµ·) | |
| (2) | ||
| template< class T > T&& forward( typename std::remove_reference<T>::type&& t ) noexcept; |
(C++11 èµ·) (C++14 å) |
|
| template< class T > constexpr T&& forward( std::remove_reference_t<T>&& t ) noexcept; |
(C++14 èµ·) | |
å½ t æ¯è½¬åå¼ç¨ï¼ä½ä¸ºå°æ cv éå®å½æ°æ¨¡æ¿å½¢åçå³å¼å¼ç¨ç彿°å®åï¼ï¼æ¤éè½½å°åæ°ä»¥å¨ä¼ éç»è°ç¨æ¹å½æ°æ¶çå¼ç±»å«è½¬åç»å¦ä¸ä¸ªå½æ°ã
ä¾å¦ï¼è¥ç¨äºå¦ä¸çå è£ å¨ï¼å模æ¿è¡¨ç°ä¸ºä¸æ¹ææè¿°ï¼
template<class T> void wrapper(T&& arg) { // arg å§ç»æ¯å·¦å¼ foo(std::forward<T>(arg)); // 转åä¸ºå·¦å¼æå³å¼ï¼ä¾èµäº T }
- è¥å¯¹
wrapper()çè°ç¨ä¼ éå³å¼std::stringï¼åæ¨å¯¼T为std::stringï¼éstd::string&æstd::string&&ï¼ä¸std::forwardç¡®ä¿å°å³å¼å¼ç¨ä¼ éç»fooã - è¥å¯¹
wrapper()çè°ç¨ä¼ é const å·¦å¼std::stringï¼åæ¨å¯¼T为const std::string&ï¼ä¸std::forwardç¡®ä¿å° const å·¦å¼å¼ç¨ä¼ éç»fooã - è¥å¯¹
wrapper()çè°ç¨ä¼ éé const å·¦å¼std::stringï¼åæ¨å¯¼T为std::string&ï¼ä¸std::forwardç¡®ä¿å°é const å·¦å¼å¼ç¨ä¼ éç»fooã
æ¤é载令转å表达å¼ï¼å¦å½æ°è°ç¨ï¼çç»æå¯è¡ï¼ç»æå¯ä»¥æ¯å³å¼æå·¦å¼ï¼å转åå¼ç¨åæ°çåå§å¼ç±»å«ã
ä¾å¦ï¼è¥å è£ å¨ä¸ä» 转åå ¶åæ°ï¼è¿å¨åæ°ä¸è°ç¨æå彿°ï¼å¹¶è½¬åå ¶ç»æï¼
// 转æ¢å è£ å¨ template<class T> void wrapper(T&& arg) { foo(forward<decltype(forward<T>(arg).get())>(forward<T>(arg).get())); }
å ¶ä¸ arg çç±»åå¯ä»¥æ¯
struct Arg { int i = 1; int get() && { return i; } // æ¤éè½½çè°ç¨ä¸ºå³å¼ int& get() & { return i; } // æ¤éè½½çè°ç¨ä¸ºå·¦å¼ };
è¯å¾è½¬åå³å¼ä¸ºå·¦å¼ï¼ä¾å¦éè¿ä»¥å·¦å¼å¼ç¨ç±»å T å®ä¾åå½¢å¼ (2) ï¼ä¼äº§çç¼è¯æ¶é误ã
注æ
转åå¼ç¨èåçç¹æ®è§åï¼ T&& ç¨ä½å½æ°åæ°ï¼è§æ¨¡æ¿å®åæ¨å¯¼ï¼å
¶ä»ç»èè§è½¬åå¼ç¨ã
åæ°
| t | - | è¦è½¬åç对象 |
è¿åå¼
static_cast<T&&>(t)
示ä¾
æ¤ç¤ºä¾æ¼ç¤ºåæ°å°ç±» T æé 彿°å®åçå®ç¾è½¬åãå¹¶å±ç¤ºåæ°å
çå®ç¾è½¬åã
#include <iostream> #include <memory> #include <utility>  struct A { A(int&& n) { std::cout << "rvalue overload, n=" << n << "\n"; } A(int& n) { std::cout << "lvalue overload, n=" << n << "\n"; } };  class B { public: template<class T1, class T2, class T3> B(T1&& t1, T2&& t2, T3&& t3) : a1_{std::forward<T1>(t1)}, a2_{std::forward<T2>(t2)}, a3_{std::forward<T3>(t3)} { }  private: A a1_, a2_, a3_; };  template<class T, class U> std::unique_ptr<T> make_unique1(U&& u) { return std::unique_ptr<T>(new T(std::forward<U>(u))); }  template<class T, class... U> std::unique_ptr<T> make_unique2(U&&... u) { return std::unique_ptr<T>(new T(std::forward<U>(u)...)); }  int main() { auto p1 = make_unique1<A>(2); // å³å¼ int i = 1; auto p2 = make_unique1<A>(i); // å·¦å¼ Â std::cout << "B\n"; auto t = make_unique2<B>(2, i, 3); }
è¾åºï¼
rvalue overload, n=2 lvalue overload, n=1 B rvalue overload, n=2 lvalue overload, n=1 rvalue overload, n=3
å¤æåº¦
常æ°
åé
| (C++11) |
è·å¾å³å¼å¼ç¨ (彿°æ¨¡æ¿) |
| (C++11) |
è¥ç§»å¨æé 彿°ä¸æåºåè·å¾å³å¼å¼ç¨ (彿°æ¨¡æ¿) |