1

好的,我开始了解右值引用的要点(我认为)。我有我正在编写的这段代码:

#include <iostream>

using namespace std;

std::string get_string()
{
  std::string str{"here is your string\n"};
  return std::move(str);  // <----- cast here?
}

int main ()
{
  std::string my_string = std::move(get_string()); // <----- or cast here?
  std::cout << my_string;
  return 0;
}

所以我有一个简单的例子,我有一个返回字符串副本的函数。我已经读过它不好(并得到了核心转储来证明它!)返回对本地临时变量的任何引用,所以我不考虑尝试。

main()我不想复制构造该字符串的分配中,我想移动构造/分配字符串以避免复制字符串太多。

  • Q1:我返回了 temp var 的“副本”,get_string()但我已将返回值转换为 rvalue-red。那是没有意义的,还是在做任何有用的事情?

  • Q2:假设 Q1 的答案是我不需要这样做。然后我是将 temp 变量的新副本移动到 中my_string,还是直接将 temp 变量移动strmy_string.

  • Q3:为了将字符串返回值存储到“外部”(在我的情况下是 main())变量中,您需要的最小副本数是多少,以及您如何做到这一点(如果我还没有实现它)?

4

2 回答 2

1

我在 get_string() 中返回了 temp var 的“副本”——但我已将返回值转换为 rvalue-red。那是没有意义的,还是在做任何有用的事情?

您不必std::move在这种情况下使用,因为值返回的局部变量会为您“隐式移动”。标准中有一个特殊的规则。在这种情况下,您的举动是悲观的,因为它可以防止 RVO(clang 警告)。


Q2:假设 Q1 的答案是我不需要这样做。然后我是将临时变量的新副本移动到 my_string 中,还是直接将临时变量 str 移动到 my_string 中。

你不需要std::move调用的结果get_string()get_string()prvalue,这意味着my_string将自动调用 的移动构造函数(C++17 之前)。在 C++17 及更高版本中,强制复制省略将确保不会发生移动/复制(使用prvalues)。


Q3:为了将字符串返回值存储到“外部”(在我的情况下是 main())变量中,您需要的最小副本数是多少,以及您如何做到这一点(如果我还没有实现它)?

取决于标准以及是否发生 RVO。如果发生 RVO,您将拥有 0 个副本和 0 个移动。如果您以 C++17 为目标并从prvalue进行初始化,则保证您有 0 个副本和 0 个移动。如果两者都没有发生,您可能只有一个动作 - 我不明白为什么这里应该出现任何副本。

于 2018-09-19T11:59:23.063 回答
1

您不需要std::move在返回值上使用局部变量。编译器会为您执行此操作

如果 expression 是一个左值表达式,它是在主体中声明的自动存储持续时间对象的(可能是带括号的)名称,或者作为最内层封闭函数或 lambda 表达式的参数,则重载决策以选择用于初始化的构造函数返回值被执行两次:第一次好像表达式是一个右值表达式(因此它可以选择移动构造函数),如果没有合适的转换可用,或者如果所选构造函数的第一个参数的类型不是一个右值引用对象的类型(可能是 cv 限定的),第二次执行重载决议,表达式被视为左值(因此它可以选择复制构造函数引用非 const)。

于 2018-09-19T12:03:33.893 回答