1.用一个对象初始化另外一个对象
int main(){
Person p1(10);
Person p2(p1);//隐式调用
Person p3 =Person(p1);//完整拷贝构造调用
Person p4 = p1;//隐式转换
return 0;
}
2.对象以值传递的方式传递给函数参数
void dowork(Person p){
//值传递,调用拷贝构造函数
}
3.函数局部对象以值传递的方式从函数返回
Person dowork1(){
Person p(30);
return p;
}
//外部调用
Person p2 = dowork1();//这一步需要拷贝
上述情形本应是dowork1函数首先创建局部对象p(调用有参构造),然后return p是编译器先在栈中创建一个临时的对象temp然后将p拷贝给temp,然后temp再拷贝给p2,应该是2次调用拷贝构造但实际只有一次。
因为:
编译器采用了RVO优化(return value optimization),该优化在调用dowork1的时候将p2的引用传递进dowork1函数,因此只发送Person p(30);这一次拷贝构造。
Release版本还会采用NRVO优化(named return value optimization),因为返回的对象是有名字的,对象不是在return语句上创建的,连拷贝构造函数都不用调用了。