局部变量是否在C++中?
Are local variables rvalues in C++?
我正在阅读Stroustrup的教科书,以了解右值和移动构造函数的概念:
右值是 - 对于第一个近似值 - 一个您无法分配的值 to,例如函数调用返回的整数和右值 引用是对其他人无法分配的内容的引用。 运算符+() 中用于向量的 res 局部变量就是一个例子。
下面列出了上面引用中提到的代码。
问题:局部变量res
可以合法地出现在赋值的左侧,例如在res=nullptr
中。为什么它被视为右值?
Vector operator+(const Vector& a, const Vector& b) {
if (a.size()!=b.size()) throw Vector_size_mismatch{};
Vector res(a.size());
for (int i=0; i!=a.size(); ++i) res[i]=a[i]+b[i];
return res;
}
不,局部变量不是右值。在您的示例中,右值是函数返回的 res 副本的临时对象(尽管可以省略复制构造函数或移动构造函数的调用)。
res
本身是函数中的一个左值。
运算符+() 中用于向量的 res 局部变量就是一个例子。
这是错误的。res
是左值的一个典型例子。我有点担心你的教科书。我们没有太多上下文,所以这可能只是一个错别字或误导性措辞的实例。但是,如果这是一个真正的错误,则表明作者对该语言的基本原理缺乏了解。
通常的经验法则是:"如果它有名字,它就不能是一个右值。如果它没有名字,它很可能是一个右值。这很有效,即使在一开始看起来令人困惑的情况下也是如此。例如:
// A temporary has no name.
// Therefore: Can call foo() with it.
foo(Object());
void foo(Object&& o)
{
// o has a name. Therefore: not an rvalue.
// Therefore: Must move.
bar(std::move(o));
}
void bar(Object&& o)
{
// ...
}
该规则也适用于代码段中的任何位置:
- 所有
.size()
返回值:无名称 -> 右值 a
、b
、res
、i
有名字 -> 而不是右值- 布尔值
i!=a.size()
:无名 -> 右值 - 临时
a[i]+b[i]
:无名称 -> 右值
只有函数的返回值更棘手,因为返回值优化 (RVO) 在这里发挥作用。这里的经验法则是:"如果你返回一个局部变量,通常不需要std::move()它。
相关文章:
- 在提升multi_index容器中,是否定义了"default index"?
- 在C++STL中是否有Polyval(Matlab函数)等价物?
- 检查输入是否不是整数或数字
- 是否可以初始化不可复制类型的成员变量(或基类)
- 在C++中,是否可以基于给定的标识符创建基类的新实例,反之亦然
- 是否可以通过C++扩展强制多个python进程共享同一内存
- 此代码是否违反一个定义规则
- 是否需要删除包含对象的"pair"?
- 是否可以从int转换为enum类类型
- 无论条件是否为true,if总是在c++中执行
- 如何找到大小'x'数组是否完全填充,在C++?
- 检查值是否在集合p1和p2中,但不在p3中
- 是否可以在编译时初始化数组,以便在运行时不会花费时间?
- 检查 std::shared_ptr<> 的当前底层类型是否为 T
- 在c++中检查长方体是否尽可能快地重叠(无迭代)
- GL_SHADERSTORAGE_BUFFER位置是否与其他着色器位置冲突
- 子目录是否继承属性,例如add_definitions,include_directories和父Cmakelist.t
- 标准是否使用多余的大括号(例如 T{{{10}}})定义列表初始化?
- C/C++预处理器是否可以检测一些编译器选项
- 是否可以用"iostream"包装现有的TCP/OOpenSSL会话