以下代码在 GCC 8.0.1 下使用或不使用用户定义的复制构造函数的行为不同:
#include <cassert>
struct S {
int i;
int *p;
S() : i(0), p(&i) {}
// S(const S &s) : i(s.i), p(&i) {} // #1
// S(const S &s) : i(s.i), p(s.p) {} // #2
// S(const S &s) = delete; // #3
};
S make_S() {return S{};}
int main()
{
S s = make_S();
assert(s.p == &s.i);
}
使用任一注释的用户定义的复制构造函数(即使使用#2,执行简单浅拷贝的构造函数),断言都不会失败,这意味着保证复制省略按预期工作。
但是,如果没有任何用户定义的复制构造函数,则断言失败,这意味着函数中的对象s
不是main
默认构造的。为什么会这样?不保证复制省略在这里执行吗?