C++的完美转发(Perfect Forwarding)是C++11引入的一个新特性,它允许函数模板将参数以原始形式(保持参数的类型和值类别)传递给其他函数。这意味着传递给完美转发的函数的参数可以是左值或右值、const或非const、引用或非引用等。
为了实现完美转发,C++引入了两个新的概念:通用引用(Universal Reference)和std::forward函数模板。
-
通用引用(Universal Reference):通用引用是一种特殊类型的引用,它可以同时绑定到左值和右值。通用引用的定义是在模板参数列表中使用T&&,其中T是一个模板参数。当T是一个左值时,T&&表示一个左值引用;当T是一个右值时,T&&表示一个右值引用。通用引用通常用于实现完美转发。
-
std::forward函数模板:std::forward函数模板用于将参数完美转发给另一个函数。它的基本语法如下:
template
constexpr T&& forward(typename std::remove_reference::type& arg) noexcept ;
std::forward函数模板接受一个通用引用参数arg,并使用std::remove_reference从中提取原始类型。然后,它根据arg的类型(左值或右值)返回相应的引用。这样,我们可以确保传递给目标函数的参数保持其原始类型和值类别。
下面是一个简单的完美转发示例:
#include
#include
void func1(int& x) {
std::cout << "func1(int&): "<< x << std::endl;
}
void func2(int&& x) {
std::cout << "func2(int&&): "<< x << std::endl;
}
template
void wrapper(T&& arg) {
func1(arg);
func2(std::forward(arg));
}
int main() {
int x = 42;
wrapper(x); // 输出:func1(int&): 42 和 func2(int&&): 42
wrapper(42); // 输出:func1(int&): 42 和 func2(int&&): 42
wrapper(std::move(x)); // 输出:func1(int&): 42 和 func2(int&&): 42
}
在这个示例中,wrapper函数模板使用完美转发将参数arg传递给func1和func2。注意,尽管arg在调用func1时是一个左值引用,但在调用func2时它是一个右值引用,这是因为std::forward会根据参数的原始类型返回相应的引用。