- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我这里有两段代码向您展示。它们是两个类,每个类都提供一个 Move 构造函数和一个返回临时值的函数。
我很困惑:在这两种情况下,我都定义了一个 move 构造函数和一个返回临时值的随机成员函数。但行为发生了变化,我的问题是为什么。
请注意,在以下示例中,运算符 << 被重载,以便打印列表(在第一种情况下)和 double 据成员(在第二种情况下)。
<小时/>调用 move 构造函数
template<typename T>
class GList
{
public:
GList() : il{ nullptr } {}
GList(const T& val) : il{ new Link<T>{ val,nullptr } } {}
GList(const GList<T>& copy) {}
GList(GList<T>&& move)
{
std::cout << "[List] Move constructor called" << std::endl;
// ... code ...
}
// HERE IS THE FUNCTION WHICH RETURNS A TEMPORARY!
GList<T> Reverse()
{
GList<T> result;
if (result.il == nullptr)
return *this;
...
...
...
return result;
}
};
int main()
{
GList<int> mylist(1);
mylist.push_head(0);
cout << mylist.Reverse();
return 0;
}
输出为:
<小时/>[List] Move constructor called
0
1
已执行复制省略
class Notemplate
{
double d;
public:
Notemplate(double val)
{
d = val;
}
Notemplate(Notemplate&& move)
{
cout << "Move Constructor" << endl;
}
Notemplate(const Notemplate& copy)
{
cout << "Copy" << endl;
}
Notemplate Redouble()
{
Notemplate example{ d*2 };
return example;
}
};
int main()
{
Notemplate my{3.14};
cout << my.Redouble();
return 0;
}
输出为:
<小时/>6.28
我期待在第二个示例中调用 move 构造函数。毕竟,该函数的逻辑是相同的:返回一个临时值。
有人能解释一下为什么没有发生吗?
如何处理复制省略?
我希望我的代码尽可能具有最佳的可移植性,我如何才能确保编译器进行此类优化?
最佳答案
在another SO answer的评论中,OP 在这里澄清了他的问题:
I heard that copy elision CAN occur even when there are more than 1 return statements. I'd like to know when a copy elision is forbidden
所以我试图在这里解决这个问题:
在以下情况下允许省略复制/move 操作(C++ 标准称为复制省略):
在具有类返回类型的函数的 return
语句中,当表达式是具有自动存储期限的非 volatile 对象的名称(其他比函数参数或由处理程序的异常声明引入的变量)具有与函数返回类型相同的类型(忽略 cv 限定),可以通过构造来省略复制/move 操作自动对象直接进入函数的返回值。
在抛出表达式中,当操作数是非 volatile 自动对象(函数或catch子句参数除外)的名称时,其范围不超出在最内层的try block 末尾(如果有的话),通过将自动对象直接构造到异常对象中,可以省略从操作数到异常对象的复制/move 操作。
当未绑定(bind)到引用的临时类对象被复制/move 到具有相同类型的类对象时(忽略 cv 限定),可以通过构造临时对象直接进入省略复制/move 的目标。
当异常处理程序的异常声明声明一个与异常对象类型相同(cv 限定除外)的对象时,可以通过将异常声明作为异常对象的别名,如果除了执行异常声明所声明的对象的构造函数和析构函数之外,程序的含义将保持不变。无法从异常对象中进行 move ,因为它始终是左值。
在所有其他情况下禁止复制省略。
函数中 return 语句的数量与复制省略的合法性没有任何关系。然而,编译器可以不执行复制省略,即使它是合法的,出于任何原因,包括返回语句的数量。
C++17 更新
现在有一些地方强制执行复制省略。如果纯右值可以直接绑定(bind)到按值函数参数、按值返回类型或命名局部变量,则在 C++17 中复制省略是强制的。这意味着编译器甚至不必检查复制或 move 构造函数。合法的 C++17:
struct X
{
X() = default;
X(const X&) = delete;
X& operator=(const X&) = delete;
};
X
foo(X)
{
return X{};
}
int
main()
{
X x = foo(X{});
}
关于c++ - move 构造函数与复制省略。哪一个会被调用?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35506708/
这是我在这里的第一篇文章,也是我第一次使用 C++。我正在查看从 Internet 获得的一些代码,但我对此有疑问。 它有一个 for 循环,像这样: for(cin >> t;t--;) 我明白它在
我目前正在开发一个网站,除其他外,该网站允许用户通过显示或隐藏他们已购买的商品来过滤市场。这适用于基本的 AJAX 调用,该调用传递可用过滤器的当前条件,然后使用 CodeIgniter 的事件记录构
我创建了一个 MWE,其中通过添加 来更改单行解决编译器错误。 以下代码无法编译: import java.util.List; public class MainClass { publi
当我想测试一些 PostgreSQL 函数 FOO() 的行为时,我发现执行类似 SELECT FOO(bar) 的查询很有用,bar一些数据我用作直接输入,而无需从真实表中SELECT。 我读到我们
在 PHP、Java、C++(以及许多其他语言)中,for 循环是这样使用的: for(int i=0;i<10;i++) 如果我已经初始化了i,我该如何省略初始化语句呢? 最佳答案 在 Java、C
我发现我们的 Android 应用出现了一个奇怪的问题,特别是在 4.4 版的 Moto X 上。 在偏好 Activity 中,所有标题的前 8 个字符都在开头用省略号截断。这也发生在溢出菜单和整个
我有一个 XElement,我必须解析它以删除结束标记中的空白。我的代码如下所示: var stringBuilder = new StringBuilder(); using (var string
假设我有两个接口(interface),X和 Y ,它们共享一些字段,但也有独立的字段: interface X { abc: number; foo: number; bar: numb
我有这个模型: var accountSchema = new mongoose.Schema({ 'seeker': { 'fullName': String,
我在 R 中运行一个具有大量时间和位置固定效应的回归。我尝试将一个漂亮的汇总表输出到 Latex 中。我从 stargazer 包切换到 huxtable,因为 stargazer 在忽略固定效果时表
假设我有一个数据框: a df a b c d 1 0 9 10 2 1 10 13 3 NA 11 14 4 3 NA 7 5 4 13 22 现在假设我
我在 R 中运行一个具有大量时间和位置固定效应的回归。我尝试将一个漂亮的汇总表输出到 Latex 中。我从 stargazer 包切换到 huxtable,因为 stargazer 在忽略固定效果时表
我想删除一个属性并返回一个新对象而不改变原始对象。 我知道我们可以像这样使用 Lodash 轻松做到这一点: const profile = { name: 'Maria', age: 30 } _.
我正在通过更改一些内容来修改 javascript 对象。当我重新创建它时,我会得到每个 key 对的索引号。 "0":{...},"1":{...}, 如何删除/省略这些 0、1、2、3、4 数字的
我正在一个非常方便且名称丰富的网站 here 上完成示例之一。 ,具体来说: func applyMutliplication(value: Int, multFunction: Int -> Int
这个问题在这里已经有了答案: Why do java source files require package declarations? (4 个答案) 关闭 6 年前。 我是 Java 的新手,
在下面的代码中: Widget makeWidget() { return Widget(); } void foo(Widget widget) { ... } foo(makeWid
这是我使用下面的调用调用的过程: CALL abc('01-04-2011','14-04-2014','28,29,36,37,38','33,34,35,41,42,43') 但问题是下面提到的查
嘿,我正在寻找一种在 yui 数据表中省略文本的好方法。我的意思是,格式化文本,使其很好地适合其单元格,并且如果文本必须被 chop ,则在其后面有一个椭圆 (...)。 我想在不使用 CSS 选择器
我有一个如下表(记录)。 ID Status AA124 Pass AA125 Pass Z_AA134 Fail Z_AA135
我是一名优秀的程序员,十分优秀!