c++11左值引用与右值引用

简介: c++11左值引用与右值引用

c++11 中做值引用与右值引用

重点理解:

  • 左值引用与右值引用的区别?
  • 左值引用与右值引有什么特例?
  • 什么是移动语义?解决哪类问题?
  • 什么是完美转发?
  • 什么是将亡值?

实现移动语义

  • 通过实现移动语义,对象赋值时,避免资源的重新分配。c++11 之前利用深拷贝解决浅拷贝问题;
  • STL中应用非常广泛;
  • 实现 std::unique_ptr 等等。

解决深拷贝问题

什么是深拷贝?什么情况下使用深拷贝? 请参考深拷贝与浅拷贝

#include <iostream>
#include <list>
#include <cstring>
class A {
public:
    A() {
        p = new int(10);
        cout << "A():p=" << p << endl;
    }
    A(const A&a) { // 深拷贝和浅拷贝
        // p= a.p;
        // a.p = nullptr;
        p = new int(10);
        memcpy(p, a.p, 10*sizeof(int));
        cout << "A(const A&):p=" << p << endl;
    }
    ~A() {
        if (p != nullptr) {
            delete []p;
            p = nullptr;
            cout << "~A()1" << endl;
        } else {
            cout << "~A()2" << endl;
        }
    }
    A(A&& a) {
        this->p = a.p;
        a.p = nullptr;
        cout << "A(A&&)" << endl;
    }
    int *p;
};
int main () {
    A a;
    A a1(a);  //调用拷贝构造函数
    cout << "a.p的地址为: " << a.p <<  " a1.p的地址为: " << a1.p << endl;
    cout << "======触发移动构造======" << endl;
    A a2(std::move(a));  //触发移动构造
    cout << "a2.p的地址为:" << a2.p << " a.p为" << a.p << endl;
}
代码执行结果:
A():p=0x862c20
A(const A&):p=0x862c40
a.p的地址为: 0x862c20 a1.p的地址为: 0x862c40
======触发移动构造======
A(A&&)
a2.p的地址为:0x862c20 a.p为0   # 对象a的资源转移到a2之后调用析构将自己销毁。
~A()1
~A()1
~A()2

STL中的应用

  • 将堆上的资源移动到容器当中
#include <iostream>
#include <list>
#include <cstring>
using std::cout;
using std::endl;
class A {
public:
    A() {
        p = new int(10);
        cout << "A():p=" << p << endl;
    }
    A(const A&a) { // 深拷贝和浅拷贝
        // p= a.p;
        // a.p = nullptr;
        p = new int(10);
        memcpy(p, a.p, 10*sizeof(int));
        cout << "A(const A&):p=" << p << endl;
    }
    ~A() {
        if (p != nullptr) {
            delete []p;
            p = nullptr;
            cout << "~A()1" << endl;
        } else {
            cout << "~A()2" << endl;
        }
    }
    A(A&& a) {
        this->p = a.p;
        a.p = nullptr;
        cout << "A(A&&)" << endl;
    }
    int *p;
};
int main () {
    list<A> alist;
    A a;
    alist.push_back(A());  //(1) A()匿名对象,右值,直接调用移动构造变成一个将亡值
    //alist.push_back(a);  //(2) 左值,具体的对象,调用拷贝构造
    //alist.push_back(std::move(a));  //(3) 将左值变为右值,调用移动构造
    auto &front = alist.front();
    cout << "front.p=" << front.p << endl;
  return 0;
}
代码执行结果:
(1) 匿名对象A()申请堆上的地址与容器中A()地址相同,说明移动资源成功
A():p=0x1f60c20       #调用了构造函数
A(A&&)          #调用了移动构造
~A()2           #析构将亡值
front.p=0x1f60c20
~A()1         #析构alist对象
(2)插入左值,调用拷贝构造函数,重新申请堆上的空间。对象地址不相同
A():p=0x7f2c20
A(const A&):p=0x7f2c60
front.p=0x7f2c60
~A()1
~A()1
(3)将左值转化为右值,调用了移动构造
A():p=0x178fc20
A(A&&)
front.p=0x178fc20
~A()2
~A()1

在unique_ptr的应

  • 明确对象只有一个拥有者;
  • unique_ptr 没有拷贝构造、没有赋值运算符、仅提供移动构造和移动赋值。

智能指针unique_ptr相关内容可以参考unique_ptr相关介绍

实现完美转发

  • 理解什么是完美?什么是不完美?
  • 怎么解决完美转发?
#include <iostream>
#include <cstring>
using std::cout;
using std::endl;
using std::forward;
void func(int &n) {
    cout << "lvalue=" << n << endl;
}
void func(int &&n) {
    cout << "rvalue=" << n << endl;
}
template<typename T>
void revoke(T&&t) {   
    func(t);  //(1)
    //func(std::forward<T>(t));  //(2)
}
int main() {
    //(1)
    int i = 10;  
    int &m = i;  //左值引用
    int &&n = 100;  //右值引用
    revoke(m);
    revoke(n);
    //(2)
    //revoke(i);
    //revoke(10);
    return 0;
}
代码执行结果:
(1)
lvalue=10
lvalue=100
(2)
lvalue=10
rvalue=10

思考

为什么没有将右值引用int &&n = 100;转发成功?

答:因为声明出来的左值引用或右值引用都是左值!

#include <iostream>
#include <cstring>
using std::cout;
using std::endl;
using std::forward;
void func(int &n) {
    cout << "lvalue=" << n << endl;
}
void func(int &&n) {
    cout << "rvalue=" << n << endl;
}
template<typename T>
void revoke(T&&t) {   
    func(std::forward<T>(t)); 
}
int main() {
    int i = 10;  
    int &m = i;  //左值引用
    int &&n = 100;  //右值引用
    revoke(m);
    revoke(n);
}
代码执行结果:
lvalue=10
lvalue=100

充电站

推荐一个零声学院免费公开课程,个人觉得老师讲得不错,分享给大家:Linux,Nginx,ZeroMQ,MySQL,Redis,fastdfs,MongoDB,ZK,流媒体,CDN,P2P,K8S,Docker,TCP/IP,协程,DPDK等技术内容,立即学习

相关文章
|
11天前
|
存储 安全 编译器
【C++航海王:追寻罗杰的编程之路】引用、内联、auto关键字、基于范围的for、指针空值nullptr
【C++航海王:追寻罗杰的编程之路】引用、内联、auto关键字、基于范围的for、指针空值nullptr
36 5
|
10天前
|
C++
C++引用
C++引用
6 1
|
3天前
|
C++
C++基础知识(二:引用和new delete)
引用是C++中的一种复合类型,它是某个已存在变量的别名,也就是说引用不是独立的实体,它只是为已存在的变量取了一个新名字。一旦引用被初始化为某个变量,就不能改变引用到另一个变量。引用的主要用途包括函数参数传递、操作符重载等,它可以避免复制大对象的开销,并且使得代码更加直观易读。
|
4天前
|
存储 自然语言处理 编译器
|
5天前
|
安全 C++
|
11天前
|
编译器 C++ 开发者
C++一分钟之-右值引用与完美转发
【6月更文挑战第25天】C++11引入的右值引用和完美转发增强了资源管理和模板灵活性。右值引用(`&&`)用于绑定临时对象,支持移动语义,减少拷贝。移动构造和赋值允许有效“窃取”资源。完美转发通过`std::forward`保持参数原样传递,适用于通用模板。常见问题包括误解右值引用只能绑定临时对象,误用`std::forward`,忽视`noexcept`和过度使用`std::move`。高效技巧涉及利用右值引用优化容器操作,使用完美转发构造函数和创建通用工厂函数。掌握这些特性能提升代码效率和泛型编程能力。
16 0
|
3天前
|
编译器 C++
【C++】string类的使用④(字符串操作String operations )
这篇博客探讨了C++ STL中`std::string`的几个关键操作,如`c_str()`和`data()`,它们分别返回指向字符串的const char*指针,前者保证以&#39;\0&#39;结尾,后者不保证。`get_allocator()`返回内存分配器,通常不直接使用。`copy()`函数用于将字符串部分复制到字符数组,不添加&#39;\0&#39;。`find()`和`rfind()`用于向前和向后搜索子串或字符。`npos`是string类中的一个常量,表示找不到匹配项时的返回值。博客通过实例展示了这些函数的用法。
|
3天前
|
存储 C++
【C++】string类的使用③(非成员函数重载Non-member function overloads)
这篇文章探讨了C++中`std::string`的`replace`和`swap`函数以及非成员函数重载。`replace`提供了多种方式替换字符串中的部分内容,包括使用字符串、子串、字符、字符数组和填充字符。`swap`函数用于交换两个`string`对象的内容,成员函数版本效率更高。非成员函数重载包括`operator+`实现字符串连接,关系运算符(如`==`, `&lt;`等)用于比较字符串,以及`swap`非成员函数。此外,还介绍了`getline`函数,用于按指定分隔符从输入流中读取字符串。文章强调了非成员函数在特定情况下的作用,并给出了多个示例代码。
|
3天前
|
C++
【C++】string类的使用④(常量成员Member constants)
C++ `std::string` 的 `find_first_of`, `find_last_of`, `find_first_not_of`, `find_last_not_of` 函数分别用于从不同方向查找目标字符或子串。它们都返回匹配位置,未找到则返回 `npos`。`substr` 用于提取子字符串,`compare` 则提供更灵活的字符串比较。`npos` 是一个表示最大值的常量,用于标记未找到匹配的情况。示例代码展示了这些函数的实际应用,如替换元音、分割路径、查找非字母字符等。