本文给出引用折叠、万能引用和完美转发的相关代码。需要注意的是万能引用(T &&arg)中匹配的类型T,要么是U,要么是U&&。forward的返回值,要么是不具名左值引用,要么是不具名右值引用。在wrapper中,对forward的调用,要么是forward<U>(U &&arg)
,要么是forward<U&>(U &arg)
。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140
| #include <iostream> #include <utility>
void some_function(int &x) { std::cout << "Left value reference: " << x << std::endl; }
void some_function(int &&x) { std::cout << "Right value reference: " << x << std::endl; }
template <typename T> constexpr T && forward(typename std::remove_reference<T>::type &t) noexcept { if (std::is_lvalue_reference<T>::value) { std::cout << "forward<U&>(U &t): " << std::endl; } else if (std::is_rvalue_reference<T>::value) { std::cout << "forward<U&&>(U &t)" << std::endl; } else { std::cout << "forward<U>(U &t)" << std::endl; } return static_cast<T &&>(t); }
template <typename T> constexpr T && forward(typename std::remove_reference<T>::type &&t) noexcept { if (std::is_lvalue_reference<T>::value) { std::cout << "forward<U&>(U &&t)" << std::endl; } else if (std::is_rvalue_reference<T>::value) { std::cout << "forward<U&&>(U &&t)" << std::endl; } else { std::cout << "forward<U>(U &&t)" << std::endl; } static_assert(!std::is_lvalue_reference<T>::value, "std::forward must not be used to convert an rvalue to an lvalue"); return static_cast<T &&>(t); }
template <typename T> void wrapper(T &&arg) {
if (std::is_lvalue_reference<T>::value) { std::cout << "wrapper<U&>(U &arg)" << std::endl; } else if (std::is_rvalue_reference<T>::value) { std::cout << "wrapper<U&&>(U &&arg)" << std::endl; } else { std::cout << "wrapper<U>(U &&arg)" << std::endl; }
some_function(forward<T>(arg)); }
void test0() { forward<int>(1); forward<int&&>(1); printf("\n"); int a = 2; forward<int>(a); forward<int&>(a); forward<int&&>(a); printf("\n");
int b = 3; forward<int>(static_cast<int&>(b)); forward<int&>(static_cast<int&>(b)); forward<int&&>(static_cast<int&>(b)); printf("\n");
b = 4; int &c = b; forward<int>(c); forward<int&>(c); forward<int&&>(c); printf("\n");
forward<int>(static_cast<int&&>(5)); forward<int&&>(static_cast<int&&>(5)); printf("\n");
int &&d = 6; forward<int>(d); forward<int&>(d); forward<int&&>(d); }
void test1() { wrapper(1); printf("\n");
int a = 2; wrapper(a); printf("\n");
a = 3; wrapper(static_cast<int &>(a)); printf("\n");
a = 4; int &b = a; wrapper(b); printf("\n");
wrapper(std::move(5)); printf("\n");
int &&c = 6; wrapper(c); }
int main() { test0(); test1(); }
|