若有时间再详细地解释一下,RVO和NRVO都做了什么, 此文只说结论。
一、RVO和NRVO优化后(NVO和NRVO是什么,自行google,此处不做解释),形如:
Test func() {
Test t;
return t;
}
Test tt = func();
只会调用一次构造函数。若想解除RVO和NRVO,编译时加上 -fno-elide-constructors 参数。
若加上-fno-elide-constructors,函数调用顺序为:
1、构造函数, func内局部变量构造
2、移动构造函数,局部变量t构造返回值的临时变量tmp,注意此处不是拷贝构造函数。
3、析构函数,局部变量t析构
4、移动构造函数,返回值的临时变量tmp构造tt
5、析构函数, 返回值的临时变量tmp析构
6、析构函数, tt析构
二、对象参数传递,若实参为局部变量则优先调用移动构造函数,若实参为全局变量或者const类型的局部变量等不会引起RVO或者NRVO的则会调用拷贝构造函数。即加上-fno-elide-constructors参数后,形如:
Test t;
Test func() {
return t;
}
Test tt = func();
以全局变量为例的函数调用顺序为:
1、构造函数, 全局变量t构造
2、拷贝构造函数,全局变量t构造返回值的临时对象tmp
3、移动构造函数, 返回值的临时对象tmp构造tt
4、析构函数, 临时对象tmp析构
5、析构函数, tt析构
6、析构函数, t析构