显然在这个问题上编译器之间存在一些混淆和差异:
根据这篇文章:
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/
我正在编写一个包含C扩展的gem。通常当我写一个gem时,我会遵循TDD的过程,我会写一个失败的规范,然后处理代码直到它通过,等等......在“ext/mygem/mygem.c”中我的C扩展和在gemspec的“扩展”中配置的有效extconf.rb,如何运行我的规范并仍然加载我的C扩展?当我更改C代码时,我需要采取哪些步骤来重新编译代码?这可能是个愚蠢的问题,但是从我的gem的开发源代码树中输入“bundleinstall”不会构建任何native扩展。当我手动运行rubyext/mygem/extconf.rb时,我确实得到了一个Makefile(在整个项目的根目录中),然后当
我的瘦服务器配置了nginx,我的ROR应用程序正在它们上运行。在我发布代码更新时运行thinrestart会给我的应用程序带来一些停机时间。我试图弄清楚如何优雅地重启正在运行的Thin实例,但找不到好的解决方案。有没有人能做到这一点? 最佳答案 #Restartjustthethinserverdescribedbythatconfigsudothin-C/etc/thin/mysite.ymlrestartNginx将继续运行并代理请求。如果您将Nginx设置为使用多个上游服务器,例如server{listen80;server
它不等于主线程的binding,这个toplevel作用域是什么?此作用域与主线程中的binding有何不同?>ruby-e'putsTOPLEVEL_BINDING===binding'false 最佳答案 事实是,TOPLEVEL_BINDING始终引用Binding的预定义全局实例,而Kernel#binding创建的新实例>Binding每次封装当前执行上下文。在顶层,它们都包含相同的绑定(bind),但它们不是同一个对象,您无法使用==或===测试它们的绑定(bind)相等性。putsTOPLEVEL_BINDINGput
我想让一个yaml对象引用另一个,如下所示:intro:"Hello,dearuser."registration:$introThanksforregistering!new_message:$introYouhaveanewmessage!上面的语法只是它如何工作的一个例子(这也是它在thiscpanmodule中的工作方式。)我正在使用标准的rubyyaml解析器。这可能吗? 最佳答案 一些yaml对象确实引用了其他对象:irb>require'yaml'#=>trueirb>str="hello"#=>"hello"ir
我想这样组织C源代码:+/||___+ext||||___+native_extension||||___+lib||||||___(Sourcefilesarekeptinhere-maycontainsub-folders)||||___native_extension.c||___native_extension.h||___extconf.rb||___+lib||||___(Rubysourcecode)||___Rakefile我无法使此设置与mkmf一起正常工作。native_extension/lib中的文件(包含在native_extension.c中)将被完全忽略。
如何将send与+=一起使用?a=20;a.send"+=",10undefinedmethod`+='for20:Fixnuma=20;a+=10=>30 最佳答案 恐怕你不能。+=不是方法,而是语法糖。参见http://www.ruby-doc.org/docs/ProgrammingRuby/html/tut_expressions.html它说Incommonwithmanyotherlanguages,Rubyhasasyntacticshortcut:a=a+2maybewrittenasa+=2.你能做的最好的事情是:
我正在尝试在Rails上安装ruby,到目前为止一切都已安装,但是当我尝试使用rakedb:create创建数据库时,我收到一个奇怪的错误:dyld:lazysymbolbindingfailed:Symbolnotfound:_mysql_get_client_infoReferencedfrom:/Library/Ruby/Gems/1.8/gems/mysql2-0.3.11/lib/mysql2/mysql2.bundleExpectedin:flatnamespacedyld:Symbolnotfound:_mysql_get_client_infoReferencedf
我对如何计算通过{%assignvar=0%}赋值的变量加一完全感到困惑。这应该是最简单的任务。到目前为止,这是我尝试过的:{%assignamount=0%}{%forvariantinproduct.variants%}{%assignamount=amount+1%}{%endfor%}Amount:{{amount}}结果总是0。也许我忽略了一些明显的东西。也许有更好的方法。我想要存档的只是获取运行的迭代次数。 最佳答案 因为{{incrementamount}}将输出您的变量值并且不会影响{%assign%}定义的变量,我
我有一个要在我的Rails3项目中使用的数组扩展方法。它应该住在哪里?我有一个应用程序/类,我最初把它放在(array_extensions.rb)中,在我的config/application.rb中我加载路径:config.autoload_paths+=%W(#{Rails.root}/应用程序/类)。但是,当我转到railsconsole时,未加载扩展。是否有一个预定义的位置可以放置我的Rails3扩展方法?或者,一种预先定义的方式来添加它们?我知道Rails有自己的数组扩展方法。我应该将我的添加到active_support/core_ext/array/conversion
我正在尝试将一个资源属性的默认值设置为另一个属性的值。我正在为我正在构建的tomcat说明书定义一个资源,其中包含以下定义。我想要可以独立设置的“名称”和“服务名称”属性。当未设置服务名称时,我希望它默认为为“名称”提供的任何内容。以下不符合我的预期:attribute:name,:kind_of=>String,:required=>true,:name_attribute=>trueattribute:service_name,:kind_of=>String,:default=>:name注意第二行末尾的“:default=>:name”。当我在Recipe的新block中引用我