【C++11特性篇】一文带小白轻松理解【万能引用(引用折叠)】&【完美转发】

简介: 【C++11特性篇】一文带小白轻松理解【万能引用(引用折叠)】&【完美转发】

一.万能引用

【1】基本概念

  • 万能引用 既可以接收左值,又可以接收右值
  • 实参是左值,他就是左值引用(引用折叠)
  • 实参是右值,他就是右值引用
  • PS:万能引用还有另一种叫法:引用折叠 ,就是当其传入参数为左值时,&&会折叠成&;当传入参数为右值时,&&不折叠照常接收

【2】在C++中的应用场景简述(代码演示)

  • 模板中的 && 不代表右值引用,而是 万能引用 ,其既能接收左值又能接收右值。
  • 模板的万能引用只是提供了能够接收同时接收左值引用和右值引用的 能力
  • 但是引用类型的唯一作用就是—— 限制了接收的类型 ,后续使用中都退化成了 左值
void Fun(int& x) { cout << "左值引用" << endl; }
void Fun(const int& x) { cout << "const 左值引用" << endl; }
void Fun(int&& x) { cout << "右值引用" << endl; }
void Fun(const int&& x) { cout << "const 右值引用" << endl; }
// 万能引用:既可以接收左值,又可以接收右值
// 实参左值,他就是左值引用(引用折叠)
// 实参右值,他就是右值引用
template<typename T>
void PerfectForward(T&& t)
{
  Fun(t);
}
int main()
{
 PerfectForward(10);           // 右值
 int a;
 PerfectForward(a);     
 return 0;
}

二.完美转发

【1】完美转发应用的引入

根据本篇博客【第四part】中的结论: 右值引用变量的属性 会被编译器识别成 左值

  • 我们希望能够在传递过程中 保持它的左值或者右值的属性, 就需要用到 std::forward 完美转发

【2】基本概念

  • std::forward 完美转发 在传参的过程中保留 对象原生类型属性,即保持它的左值或者右值的属性

【3】在C++中的应用场景简述(代码演示)

void Fun(int& x) { cout << "左值引用" << endl; }
void Fun(const int& x) { cout << "const 左值引用" << endl; }
void Fun(int&& x) { cout << "右值引用" << endl; }
void Fun(const int&& x) { cout << "const 右值引用" << endl; }
template<typename T>
void PerfectForward(T&& t)
{
    // std::forward<T>(t)在传参的过程中保持了t的原生类型属性
  // 完美转发,t是左值引用,保持左值属性
  // 完美转发,t是右值引用,保持右值属性
    Fun(std::forward<T>(t));
}
int main()
{
 PerfectForward(10);           // 右值
 int a;
 PerfectForward(a);     
 return 0;
}

三.完美转发实际中的使用场景

【1】希望传入函数的右值能够保留右值走【移动构造】而不是【拷贝构造】

具体情景演示如下所示:

  • 比如在string的push_back函数中,以往,它是传左值,左值引用调用拷贝构造;
  • 而在C++11中,容器都支持了【移动构造】,所以我们有时会传右值,希望它能够走移动构造
  • 但问题是:右值引用变量的属性会被编译器识别成左值! 如下图中传到push_back函数中的val已经是左值了,想走匹配的右值引用实现移动构造就成了问题
  • 而我们此时给它用上【完美转发】在传参的过程中保留对象原生类型属性; val的属性仍然保存为右值,就可以正常走右值引用实现移动构造了

四.关于【左值引用】【右值引用】易混淆的知识点

【1】结论:右值引用变量的属性会被编译器识别成左值

  • 右值引用变量的属性会被编译器识别成左值
  • 否则在移动构造的场景下无法完成 资源转移(移动构造),必须要修改

【2】结论的证明(代码演示)

  • 我们可以观察下面代码,证明该结论:
int main()
{
    int a;
  int& r = a;
  int&& rr = move(a);//std::move()函数位于头文件中,该函数名字具有迷惑性,它并不搬移任何东西
                      //唯一的功能就是将一个左值强制转化为右值引用,然后实现移动语义
  cout << &r << endl;
  cout << &rr << endl;  //我们知道右值不能取地址,不能被修改,而这里都能正常打印
                       //证明结论:右值引用变量的属性会被编译器识别成左值
 return 0;
}


相关文章
|
1月前
|
编译器 程序员 定位技术
C++ 20新特性之Concepts
在C++ 20之前,我们在编写泛型代码时,模板参数的约束往往通过复杂的SFINAE(Substitution Failure Is Not An Error)策略或繁琐的Traits类来实现。这不仅难以阅读,也非常容易出错,导致很多程序员在提及泛型编程时,总是心有余悸、脊背发凉。 在没有引入Concepts之前,我们只能依靠经验和技巧来解读编译器给出的错误信息,很容易陷入“类型迷路”。这就好比在没有GPS导航的年代,我们依靠复杂的地图和模糊的方向指示去一个陌生的地点,很容易迷路。而Concepts的引入,就像是给C++的模板系统安装了一个GPS导航仪
103 59
|
1月前
|
存储 编译器 C++
【C++】面向对象编程的三大特性:深入解析多态机制(三)
【C++】面向对象编程的三大特性:深入解析多态机制
|
1月前
|
存储 编译器 C++
【C++】面向对象编程的三大特性:深入解析多态机制(二)
【C++】面向对象编程的三大特性:深入解析多态机制
|
1月前
|
编译器 C++
【C++】面向对象编程的三大特性:深入解析多态机制(一)
【C++】面向对象编程的三大特性:深入解析多态机制
|
1月前
|
存储 安全 编译器
【C++】C++特性揭秘:引用与内联函数 | auto关键字与for循环 | 指针空值(一)
【C++】C++特性揭秘:引用与内联函数 | auto关键字与for循环 | 指针空值
|
26天前
|
C++
C++ 20新特性之结构化绑定
在C++ 20出现之前,当我们需要访问一个结构体或类的多个成员时,通常使用.或->操作符。对于复杂的数据结构,这种访问方式往往会显得冗长,也难以理解。C++ 20中引入的结构化绑定允许我们直接从一个聚合类型(比如:tuple、struct、class等)中提取出多个成员,并为它们分别命名。这一特性大大简化了对复杂数据结构的访问方式,使代码更加清晰、易读。
32 0
|
2月前
|
编译器 C++ 计算机视觉
C++ 11新特性之完美转发
C++ 11新特性之完美转发
49 4
|
2月前
|
Java C# C++
C++ 11新特性之语法甜点1
C++ 11新特性之语法甜点1
31 4
|
2月前
|
安全 程序员 编译器
C++ 11新特性之auto和decltype
C++ 11新特性之auto和decltype
38 3
|
2月前
|
设计模式 缓存 安全
C++ 11新特性之week_ptr
C++ 11新特性之week_ptr
35 2