- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
显然在这个问题上编译器之间存在一些混淆和差异:
根据这篇文章:
What are rvalues, lvalues, xvalues, glvalues, and prvalues?
Xvalues 是 rvalues(连同 prvalues)并且标准说:
The second context is when a reference is bound to a temporary. The temporary to which the reference is bound or the temporary that is the complete object of a subobject to which the reference is bound persists for the lifetime of the reference except:
然而,有些帖子对此提出异议:
Do rvalue references allow dangling references?
有人可以澄清一下这个问题吗? MSVC 一次就对了吗?
最佳答案
Xvalues 可能是 rvalues,但这并不意味着它们是临时值。临时对象的生命周期延长来自于它们是临时对象,而不是它们的值(value)类别。
我故意不去了解运算符的处理顺序(这样,我强制自己编写的代码要么使用显式括号,要么不关心顺序)。您在此处复制的特定 adder
示例代码确实关心:
template <class T>
struct addable
{
friend T operator +( const T& lhs, const T& rhs )
{
return std::move(T(lhs) += rhs);
}
friend T operator +( const T& lhs, T&& rhs )
{
return std::move(T(lhs) += std::move(rhs));
}
friend T&& operator +( T&& lhs, const T& rhs )
{
return std::move(lhs += rhs);
}
friend T&& operator +( T&& lhs, T&& rhs )
{
return std::move(lhs += std::move(rhs));
}
};
如果 +
运算符是从右到左执行的,那么 t1 + t2 + t3
将得出 t1 + (t2 + t3)
。 t2 + t3
将调用第一个重载,从而产生一个临时,从而产生 t1 + temp
。由于临时对象将优先绑定(bind)到右值引用,该表达式将调用第二个重载,它也返回一个临时对象。
但是,如果 +
运算符从左到右工作,那么您将得到 (t1 + t2) + t3
。这给了我们 temp + t1
,这会导致问题。它将调用第三个重载。该函数的 lhs
参数是一个 T&&
,对临时文件的引用。您返回相同的引用。这意味着您已经返回了对临时文件的引用。但是 C++ 并不知道这一点;它只知道您正在返回对某物的引用。
然而,在评估最终表达式(对新变量的赋值,值类型或引用类型)后,该“某物”将被销毁。请记住:C++ 不知道此函数将返回对其第一个参数的引用。因此它无法知道传递给函数操作数的临时对象的生命周期需要延长到存储返回引用的生命周期。
顺便说一下,这就是为什么表达式树在 auto
和类似潜伏的情况下会很危险。因为创建的内部临时对象不能被存储在各种对象中的新临时对象或引用保留。 C++ 只是没有办法做到这一点。
所以谁是对的取决于运算符的解析顺序。但是,我更喜欢我的解决方案:不要依赖语言的这些角落,而是绕过它们。停止从这些重载返回 T&&
并将值移动到临时值。这样,它就可以保证正常工作,并且您不必经常检查标准来确保您的代码正常工作。
另外,顺便说一句,我认为 operator+ 实际修改其中一个参数有些粗鲁。
但是,如果您非要知道谁是对的,那就是 GCC。来自第 5.7 节,p1:
The additive operators + and - group left-to-right.
所以是的,它不应该起作用。
注意:Visual Studio 允许 T &r2 = t1 + t2 + t3;
编译为(非常烦人的)语言扩展。您应该希望从中得到警告。
关于c++ - xvalue 的生命周期绑定(bind)到引用扩展与否?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/17768746/
根据我的理解,下面的代码应该调用 Test 类的 move 构造函数,因为这个函数按值返回,这意味着表达式 GetTestObj() 应该是右值,xvalues 是隐式的 move 但为什么这段代码调
所以我读了这个answer因为我对何时将值视为 xvalue 感到困惑,例如当值即将到期/接近其生命周期结束时。可悲的是,我仍然很困惑。 无论如何,引文包括: a class member acces
我阅读了关于xvalue 的标准。它与 rvalue reference 表达式密切相关。但是在编程的时候,实在找不到需要xvalue的场景。 例如:函数 myClass&& f() 返回一个 rva
从 5.2.1.1 开始: The expression E1[E2] is identical (by definition) to *((E1)+(E2)) [...] except that i
根据我对此发表的评论: passing std::vector to constructor and move semantics以下代码中是否需要std::move,以确保返回值是xvalue? s
我想知道是否有人可以讲述或解释一些 xvalues、glvalues 和 prvalues 的现实生活示例?我读过一个类似的问题: What are rvalues, lvalues, xvalues
我正在尝试理解 C++11 的概念。 我所说的标准草案: An xvalue (an “eXpiring” value) also refers to an object, usually near
here弄清楚 “具有身份”:地址,一个指针,用户可以判断两个拷贝是否相同。 xvalue:具有标识并且可以从中移动(例如将左值转换为右值引用的结果。 here弄清楚 以下表达式是 xvalue 表达
这个问题在这里已经有了答案: 关闭 10 年前。 Possible Duplicate: What constitutes a valid state for a “moved from” obje
假设我们有一个函数: struct A { int m; }; A&& f(); 据我所知的表达方式: f(); f().m; 都是xvalue。但为什么?为什么他们不是prvalue?我有点
我们有以下类型 X 和函数 f: struct X { ... }; X f() { ... }; 现在考虑另一个函数 g 的三个替代定义: (1) void g() { X x = f();
在cpprefernce section: Value categories ,它指出“对象表达式的成员,其中 a 是右值,m 是非引用类型的非静态数据成员”是一个 xvalue。在标准中(我在:N4
根据这份文件: http://www.stroustrup.com/terminology.pdf l 值具有同一性且不可移动。 公关值是可移动的,但没有身份。 x 值具有同一性并且是可移动的。 关于
C++ 标准对“xvalues”的描述如下(N4762 § 7.2.1.4): An expression is an xvalue if it is: - . . . - a class membe
C++11 引入了新的值类别,其中之一是 xvalue . 是explained由 Stroustrup 描述为类似(im 类别):“它是一个值,具有身份,但可以从中 move ”。 另一个来源, c
我目前正在写我的学位论文,它还涉及对 C++11 背后理论的一些解释,这真的很好,因为 C++ 是我选择的编程语言,而且该标准或多或少是免费提供的 (N3337)让自己迷路。 然而,我在尝试准确详细地
我是 C++ 的新手,这是我的第一个问题,所以请多多包涵……我已经阅读左值和右值有一段时间了,我想我理解了其中的大部分内容,但仍有一些内容令人困惑我 ... 所以我的问题会很具体 右值引用被认为是左值
有点令人惊讶(对我来说),以下两个程序编译成不同的输出,后一个具有更好的性能(用 gcc 和 clang 测试): #include int main() { std::vector a(2
xvalue的定义如下: — An xvalue (an “eXpiring” value) also refers to an object, usually near the end of its
我按照教程进行操作:http://www.amcharts.com/tutorials/detecting-at-what-value-mouse-pointer-is/但我没有得到正确的值。这是代码
我是一名优秀的程序员,十分优秀!