C++11 右值引用和移动语义(一)

简介: C++11 右值引用和移动语义

基本概念

左值和右值

左值是什么?


左值是一个表示数据的表达式 比如说变量名还有解引用的指针

他有下面的两个特点

  • 左值可以被取地址 可以被修改 (被const修饰的左值除外)
  • 左值既可以出现在表达式的左边 也可以出现在表达式的右边


下面给出一段代码示例

// 以下的p b c *p都是左值
  int* p = new int(0);
  int b = 1;
  const int c = 2;
右值是什么?

右值也是一个表示数据的表达式 比如说字符常量 表达式的返回值 函数的返回值(不能是左值引用返回)等等


他有下面的两个特点

  • 右值不能被取地址 也不能被修改
  • 右值只能出现在表达式的右边 不能出现在表达式左边


下面给出一段代码示例

  double x = 1.1;
  double y = 2.2;
  // 以下是常见的右值
  10;
  x + y;
  fmin(x, y);
  // 以下为错误示例 (右值不能出现在赋值符号的左边)
  // 10 = 1;
  // x + y = 1;
  // fmin(x, y) = 1;


  • 右值的本质是临时变量或常量 比如说上面代码中的10就是常量 x+y 和 fmin(x, y)就是临时变量
  • 临时变量和常量并没有被储存起来 这也是为什么它们不能被取地址的原因
  • 这里说的函数的返回值是右值 指的是传值返回的函数 因为传值返回的函数在返回对象的时候返回的是对象的拷贝 是一份临时变量


这里有一点需要特别说明的是 对于左值引用返回的函数来说 它们的返回值是左值

下面是一段代码示例

  class string
  {
  public:
    // 重载方括号运算符
    char& operator[](size_t i)
    {
      assert(i < _size);
      return _str[i];
    }
    // ..
  private:
    char* _str;
    size_t _size;
  };

我们这里使用【】运算符重载返回字符串中一个字符的引用 因为他要支持读写 所以采用的是左值引用返回

之所以说这里的返回值是一个左值 是因为返回的字符是被储存起来的 他是string类中_str数组中的某个字符 所以说它是可以被取地址的


左值引用和右值引用

传统的C++语法中就有引用的语法 而在C++11中更新了右值引用的语法

为了进行区分 我们将C++11之前的引用叫做左值引用 将C++11之后更新的引用叫做右值引用

不论是左值引用还是右值引用 它们的本质都是 “取别名”

左值引用


左值引用就是对于左值的引用 即对左值取别名 通过&来声明

下面是一段代码示例

  // 以下的p b c *p都是左值
  int* p = new int(0);
  int b = 1;
  const int c = 2;
  // 以下几个是对上面左值的左值引用
  int*& rp = p;
  int& rb = b;
  const int& rc = c;
  int& pvalue = *p;

右值引用

右值引用就是对于右值的引用 即对右值取别名 通过&&来声明

下面是一段代码示例

  double x = 1.1;
  double y = 2.2;
  // 以下是常见的右值
  10;
  x + y;
  fmin(x, y);
  // 以下为右值引用代码 
  int&& rr1 = 10;
  int&& rr2 = x + y;
  int&& rr3 = fmin(x, y);


这里有一点需要特别注意的 右值是不能取地址的 但是我们给右值取别名之后它会被储存到特定的位置 而此时这个右值就可以被取地址也可以被修改了

如果我们不想让右值可以被取地址和修改我们可以在前面使用const来修饰右值引用

下面是一段代码示例

int&& rr2 = x + y;
  const int&& rr3 = fmin(x, y);
  rr2 = 20; // 修改成功
  rr3 = 30; // 因为被const修饰了 所以值不能被修改

左值引用可以引用右值嘛

  • 左值引用不能引用右值 这里涉及到一个权限放大的问题 左值是可以被修改的 而右值是不可以被修改的
  • 如果想要用左值引用来引用右值 需要用到const关键字来修饰左值引用 因为经过const修饰后左值引用就没有修改的权限了


因此const既可以引用左值也可以引用右值

template<class T>
void func(const T& val)
{
  cout << val << endl;
}
int main()
{
  string s("hello");
  func(s); //s为变量 为左值
  func("world"); // "world"是常量 为右值
  return 0;
}


右值引用可以引用左值嘛

  • 右值引用只能引用右值 不能引用左值
  • 如果想要用右值引用来引用左值 需要用到move函数


move函数是C++11标准提供的一个函数 被move后的左值能够被右值引用引用

我们可以发现没有被move函数修饰之前是不可以被右值引用的

0422f935e7eb4903ab0db8d1570ea98f.png

但是被修饰之后就可以引用了

268f14914a314c10ab84c85f2d54a834.png

相关文章
|
2月前
|
存储 安全 C++
【C++11】右值引用
C++11引入的右值引用(rvalue references)是现代C++的重要特性,允许更高效地处理临时对象,避免不必要的拷贝,提升性能。右值引用与移动语义(move semantics)和完美转发(perfect forwarding)紧密相关,通过移动构造函数和移动赋值运算符,实现了资源的直接转移,提高了大对象和动态资源管理的效率。同时,完美转发技术通过模板参数完美地转发函数参数,保持参数的原始类型,进一步优化了代码性能。
45 2
|
4月前
|
编译器 C++
C++ 11新特性之右值引用
C++ 11新特性之右值引用
59 1
|
8月前
|
编译器 C语言 C++
从C语言到C++_33(C++11_上)initializer_list+右值引用+完美转发+移动构造/赋值(中)
从C语言到C++_33(C++11_上)initializer_list+右值引用+完美转发+移动构造/赋值
48 1
从C语言到C++_33(C++11_上)initializer_list+右值引用+完美转发+移动构造/赋值(中)
|
8月前
|
存储 安全 C语言
从C语言到C++_33(C++11_上)initializer_list+右值引用+完美转发+移动构造/赋值(上)
从C语言到C++_33(C++11_上)initializer_list+右值引用+完美转发+移动构造/赋值
46 2
|
8月前
|
编译器 C语言 C++
从C语言到C++_33(C++11_上)initializer_list+右值引用+完美转发+移动构造/赋值(下)
从C语言到C++_33(C++11_上)initializer_list+右值引用+完美转发+移动构造/赋值
55 1
|
7月前
|
编译器 C++ 开发者
C++一分钟之-右值引用与完美转发
【6月更文挑战第25天】C++11引入的右值引用和完美转发增强了资源管理和模板灵活性。右值引用(`&&`)用于绑定临时对象,支持移动语义,减少拷贝。移动构造和赋值允许有效“窃取”资源。完美转发通过`std::forward`保持参数原样传递,适用于通用模板。常见问题包括误解右值引用只能绑定临时对象,误用`std::forward`,忽视`noexcept`和过度使用`std::move`。高效技巧涉及利用右值引用优化容器操作,使用完美转发构造函数和创建通用工厂函数。掌握这些特性能提升代码效率和泛型编程能力。
65 0
|
11天前
|
C++ 芯片
【C++面向对象——类与对象】Computer类(头歌实践教学平台习题)【合集】
声明一个简单的Computer类,含有数据成员芯片(cpu)、内存(ram)、光驱(cdrom)等等,以及两个公有成员函数run、stop。只能在类的内部访问。这是一种数据隐藏的机制,用于保护类的数据不被外部随意修改。根据提示,在右侧编辑器补充代码,平台会对你编写的代码进行测试。成员可以在派生类(继承该类的子类)中访问。成员,在类的外部不能直接访问。可以在类的外部直接访问。为了完成本关任务,你需要掌握。
51 18
|
11天前
|
存储 编译器 数据安全/隐私保护
【C++面向对象——类与对象】CPU类(头歌实践教学平台习题)【合集】
声明一个CPU类,包含等级(rank)、频率(frequency)、电压(voltage)等属性,以及两个公有成员函数run、stop。根据提示,在右侧编辑器补充代码,平台会对你编写的代码进行测试。​ 相关知识 类的声明和使用。 类的声明和对象的声明。 构造函数和析构函数的执行。 一、类的声明和使用 1.类的声明基础 在C++中,类是创建对象的蓝图。类的声明定义了类的成员,包括数据成员(变量)和成员函数(方法)。一个简单的类声明示例如下: classMyClass{ public: int
37 13
|
11天前
|
编译器 数据安全/隐私保护 C++
【C++面向对象——继承与派生】派生类的应用(头歌实践教学平台习题)【合集】
本实验旨在学习类的继承关系、不同继承方式下的访问控制及利用虚基类解决二义性问题。主要内容包括: 1. **类的继承关系基础概念**:介绍继承的定义及声明派生类的语法。 2. **不同继承方式下对基类成员的访问控制**:详细说明`public`、`private`和`protected`继承方式对基类成员的访问权限影响。 3. **利用虚基类解决二义性问题**:解释多继承中可能出现的二义性及其解决方案——虚基类。 实验任务要求从`people`类派生出`student`、`teacher`、`graduate`和`TA`类,添加特定属性并测试这些类的功能。最终通过创建教师和助教实例,验证代码
37 5
|
11天前
|
存储 算法 搜索推荐
【C++面向对象——群体类和群体数据的组织】实现含排序功能的数组类(头歌实践教学平台习题)【合集】
1. **相关排序和查找算法的原理**:介绍直接插入排序、直接选择排序、冒泡排序和顺序查找的基本原理及其实现代码。 2. **C++ 类与成员函数的定义**:讲解如何定义`Array`类,包括类的声明和实现,以及成员函数的定义与调用。 3. **数组作为类的成员变量的处理**:探讨内存管理和正确访问数组元素的方法,确保在类中正确使用动态分配的数组。 4. **函数参数传递与返回值处理**:解释排序和查找函数的参数传递方式及返回值处理,确保函数功能正确实现。 通过掌握这些知识,可以顺利地将排序和查找算法封装到`Array`类中,并进行测试验证。编程要求是在右侧编辑器补充代码以实现三种排序算法
27 5