在 C++ 中,左值引用允许我们绑定到左值(具有明确存储位置的对象),从而可以用于函数参数的传值和返回值。为了区分左值引用,我们可以使用以下方法:
&
符号:在类型前面加上 &
符号表示该类型是一个左值引用。例如,int&
是一个左值引用类型。int x = 42;
int& ref_x = x; // ref_x 是一个左值引用,它绑定到 x
template <typename T>
void foo(T&& arg) {
// 通用实现,适用于左值和非左值引用
}
template <typename T>
void foo(T& arg) {
// 针对左值引用的特化实现
}
在这个例子中,当传递一个左值时,编译器会自动选择针对左值引用的特化版本。当传递一个右值时,编译器会选择通用实现。
std::move
:std::move
可以将右值转换为左值引用,从而触发左值引用的特化实现。例如:#include <iostream>
#include <utility>
template <typename T>
void foo(T&& arg) {
// 通用实现,适用于左值和非左值引用
std::cout << "通用实现" << std::endl;
}
template <typename T>
void foo(T& arg) {
// 针对左值引用的特化实现
std::cout << "左值引用特化实现" << std::endl;
}
int main() {
int x = 42;
foo(x); // 输出 "左值引用特化实现"
foo(42); // 输出 "通用实现"
foo(std::move(x)); // 输出 "通用实现"
return 0;
}
在这个例子中,当我们传递一个左值时,编译器会选择针对左值引用的特化实现。当我们传递一个右值时,编译器会选择通用实现。注意,在调用 foo(std::move(x))
时,x
的值已经被移动,因此再次传递 x
会导致未定义行为。