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

相关文章
|
3月前
|
存储 安全 C++
【C++11】右值引用
C++11引入的右值引用(rvalue references)是现代C++的重要特性,允许更高效地处理临时对象,避免不必要的拷贝,提升性能。右值引用与移动语义(move semantics)和完美转发(perfect forwarding)紧密相关,通过移动构造函数和移动赋值运算符,实现了资源的直接转移,提高了大对象和动态资源管理的效率。同时,完美转发技术通过模板参数完美地转发函数参数,保持参数的原始类型,进一步优化了代码性能。
56 2
|
5月前
|
编译器 C++
C++ 11新特性之右值引用
C++ 11新特性之右值引用
69 1
|
9月前
|
编译器 C语言 C++
从C语言到C++_33(C++11_上)initializer_list+右值引用+完美转发+移动构造/赋值(中)
从C语言到C++_33(C++11_上)initializer_list+右值引用+完美转发+移动构造/赋值
58 1
从C语言到C++_33(C++11_上)initializer_list+右值引用+完美转发+移动构造/赋值(中)
|
9月前
|
存储 安全 C语言
从C语言到C++_33(C++11_上)initializer_list+右值引用+完美转发+移动构造/赋值(上)
从C语言到C++_33(C++11_上)initializer_list+右值引用+完美转发+移动构造/赋值
49 2
|
9月前
|
编译器 C语言 C++
从C语言到C++_33(C++11_上)initializer_list+右值引用+完美转发+移动构造/赋值(下)
从C语言到C++_33(C++11_上)initializer_list+右值引用+完美转发+移动构造/赋值
62 1
|
8月前
|
编译器 C++ 开发者
C++一分钟之-右值引用与完美转发
【6月更文挑战第25天】C++11引入的右值引用和完美转发增强了资源管理和模板灵活性。右值引用(`&&`)用于绑定临时对象,支持移动语义,减少拷贝。移动构造和赋值允许有效“窃取”资源。完美转发通过`std::forward`保持参数原样传递,适用于通用模板。常见问题包括误解右值引用只能绑定临时对象,误用`std::forward`,忽视`noexcept`和过度使用`std::move`。高效技巧涉及利用右值引用优化容器操作,使用完美转发构造函数和创建通用工厂函数。掌握这些特性能提升代码效率和泛型编程能力。
74 0
|
5天前
|
编译器 C语言 C++
类和对象的简述(c++篇)
类和对象的简述(c++篇)
|
2天前
|
编译器 C++ 开发者
【C++篇】深度解析类与对象(下)
在上一篇博客中,我们学习了C++的基础类与对象概念,包括类的定义、对象的使用和构造函数的作用。在这一篇,我们将深入探讨C++类的一些重要特性,如构造函数的高级用法、类型转换、static成员、友元、内部类、匿名对象,以及对象拷贝优化等。这些内容可以帮助你更好地理解和应用面向对象编程的核心理念,提升代码的健壮性、灵活性和可维护性。
|
2天前
|
安全 编译器 C语言
【C++篇】深度解析类与对象(中)
在上一篇博客中,我们学习了C++类与对象的基础内容。这一次,我们将深入探讨C++类的关键特性,包括构造函数、析构函数、拷贝构造函数、赋值运算符重载、以及取地址运算符的重载。这些内容是理解面向对象编程的关键,也帮助我们更好地掌握C++内存管理的细节和编码的高级技巧。
|
2天前
|
存储 程序员 C语言
【C++篇】深度解析类与对象(上)
在C++中,类和对象是面向对象编程的基础组成部分。通过类,程序员可以对现实世界的实体进行模拟和抽象。类的基本概念包括成员变量、成员函数、访问控制等。本篇博客将介绍C++类与对象的基础知识,为后续学习打下良好的基础。