- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
这个非常简单的代码是允许的:
class s
{
public:
};
const s& tt = std::move(s()); // Also valid without 'const'
但现在我想知道为什么它被允许..
首先我们使用 std::move
并将右值(临时)标记为右值引用,但为什么左值引用可以绑定(bind)到右值引用?
是因为右值引用也是右值吗?
导致左值引用绑定(bind)到右值引用的基本原理(或标准引号)是什么?可以做到我明白了。
编辑:msvc2015 允许非 const 左值引用绑定(bind)到右值引用,问题仍然是 const 左值引用绑定(bind)到右值引用。抱歉,我应该指定我使用的编译器。
最佳答案
作为标准转换之一(C++ 标准第 4 章),右值引用被隐式转换为右值(更具体地说,转换为 xvalue):
The effect of any implicit conversion is the same as performing the corresponding declaration and initialization and then using the temporary variable as the result of the conversion. The result is an lvalue if T is an lvalue reference type or an rvalue reference to function type (8.3.2), an xvalue if T is an rvalue reference to object type, and a prvalue otherwise
右值(包括 xvalue)可以绑定(bind)到 const
左值引用,这样您就可以将临时值传递给具有此类参数的函数:
void foo(const bar &a);
// ...
foo(bar());
(临时值是右值;在本例中,bar()
的结果是右值)。没有理由不允许允许这样做,因为临时对象总是与包含表达式(在本例中为函数调用)一样长,因此它不会在 foo 中创建悬空引用
。
这意味着总是可以将函数 fun(bar)
的签名调整为 fun(const bar &)
- 当然也可以相应地更改实现! - 因为临时参数仍然会被接受,并且从调用者的角度来看语义应该是相同的; const
表示该对象不会被修改,如果通过复制传递也是如此。
不允许使用非const
引用;一个实际的原因是因为它们暗示应该以某种有意义的方式修改该值,如果该值是临时的,它将丢失。但是,如果您真的想这样做,可以将右值转换为左值,但有一些注意事项,如 this answer 中所述。另一个问题。
允许右值绑定(bind)到 const
左值引用,除了允许通过引用传递临时参数外,对于不知道确切参数类型但又希望允许的情况也有好处如果可能的话移动语义。假设我正在调用一个可以定义为 foo2(const bar &)
或 foo2(bar)
的函数,在前一种情况下,它可能有也可能没有重载foo2(bar &&)
,并且我希望尽可能使用移动语义(假设 foo2(bar &&)
重载将在其实现中使用移动语义) ;我可以安全地使用 std::move
来创建右值,因为它适用于任何一种情况。这个例子可能看起来有点做作,但这是编写模板时经常会出现的事情。在代码中:
bar bb = bar();
foo2(std::move(bb));
// above is legal if foo2 is declared as either:
// foo2(bar)
// foo2(const bar &)
// and in latter case calls overload foo2(bar &&) if it is defined.
在涉及临时对象的其他右值到左值引用赋值的情况下,临时对象的生命周期会延长到引用的生命周期,因此即使在参数传递以外的上下文中也不会创建悬挂引用:
const bar &b = bar(); // temporary lifetime is extended
在上面,bar
对象不会被销毁,直到引用 b
超出范围。
关于c++ - 为什么 const/nonconst 左值引用都绑定(bind)到右值引用?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/40909095/
这个问题在这里已经有了答案: 关闭 10 年前。 Possible Duplicate: template pass by value or const reference or…? 以下对于将函数
我用相同的参数列表重载了一个运算符两次。但返回类型不同: T& operator()(par_list){blablabla} const T& operator()(par_list){bla
假设我有实现接口(interface) I 的 Activity A。我的 ViewModel 类 (VM) 持有对实现接口(interface) I 的对象的引用: class A extends
PHP 如何解释 &$this ?为什么允许? 我遇到了以下问题,这看起来像是 PHP 7.1 和 7.2 中的错误。它与 &$this 引用和跨命名空间调用以及 call_user_func_arr
谁能解释一下下面“&”的作用: class TEST { } $abc =& new TEST(); 我知道这是引用。但是有人可以说明我为什么以及什么时候需要这样的东西吗?或者给我指向一个对此有很好解
引用变量是一个别名,也就是说,它是某个已存在变量的另一个名字。一旦把引用初始化为某个变量,就可以使用该引用名称或变量名称来指向变量。 C++ 引用 vs 指针 引用很容易与指针混淆,它们之间有三
目录 引言 背景 结论 引言 我选择写C++中的引用是因为我感觉大多数人误解了引用。而我之所以有这个感受是因为我主持过很多C++的面试,并且我很少
Perl 中的引用是指一个标量类型可以指向变量、数组、哈希表(也叫关联数组)甚至函数,可以应用在程序的任何地方 创建引用 定义变量的时候,在变量名前面加个 \,就得到了这个变量的一个引用 $sc
我编写了一个将从主脚本加载的 Perl 模块。该模块使用在主脚本中定义的子程序(我不是维护者)。 对于主脚本中的一个子例程,需要扩展,但我不想修补主脚本。相反,我想覆盖我的模块中的函数并保存对原始子例
我花了几个小时试图掌握 F# Quotations,但我遇到了一些障碍。我的要求是从可区分的联合类型中取出简单的函数(只是整数、+、-、/、*)并生成一个表达式树,最终将用于生成 C 代码。我知道使用
很多时候,问题(尤其是那些标记为 regex 的问题)询问验证密码的方法。似乎用户通常会寻求密码验证方法,包括确保密码包含特定字符、匹配特定模式和/或遵守最少字符数。这篇文章旨在帮助用户找到合适的密码
我想通过 MIN 函数内的地址(例如,C800)引用包含文本的最后一个单元格。你能帮忙吗? Sub Set_Formula() ' ----------------------------- Dim
使用常规的 for 循环,我可以做类似的事情: for (let i = 0; i < objects.length; i++) { delete objects[i]; } 常规的 for-
在 Cucumber 中,您定义了定义 BDD 语法的步骤;例如,您的测试可能有: When I navigate to step 3 然后你可以定义一个步骤: When /^I navigate t
这是什么UnaryExpression的目的,以及应该怎样使用? 最佳答案 它需要一个 Expression对象并用另一个 Expression 包裹它.例如,如果您有一个用于 lambda 的表达式
给出以下内容 $("#identifier div:first, #idetifier2").fadeOut(300,function() { // I need to reference jus
我不知道我要找的东西的正确术语,但我要找的是一个完整的引用,可以放在双引号之间的语句,比如 *, node()、@* 以及所有列出的 here加上任何其他存在的。 我链接到的答案提供了一些细节,但还
This question's answers are a community effort。编辑现有答案以改善此职位。它当前不接受新的答案或互动。 这是什么? 这是常见问答的集合。这也是一个社区Wi
Closed. This question does not meet Stack Overflow guidelines。它当前不接受答案。 想改善这个问题吗?更新问题,以便将其作为on-topic
考虑下一个代码: fn get_ref(slice: &'a Vec, f: fn(&'a Vec) -> R) -> R where R: 'a, { f(slice) } fn m
我是一名优秀的程序员,十分优秀!