首页 > 代码库 > C++什么时候调用拷贝构造函数和什么时候调用构造函数
C++什么时候调用拷贝构造函数和什么时候调用构造函数
拷贝构造函数 都是在什么情况下调用???
/* ** 什么时候调用拷贝构造函数 ** */ #include <iostream> using namespace std; //日期类 class Date{ public: int year,month,day; Date(){//无参构造 cout << "日期类的构造函数" << endl; } ~Date(){ cout << "日期的析构函数" << endl; } #pragma 拷贝构造函数里面的参数类型必须是引用 Date (const Date& d3){ this->year = d3.year; this->month = d3.month; this->day = d3.day; cout << "拷贝构造函数Date (Date& d3)" << endl; } }; void fa(Date d){//在这个参数可以看出会创建一个新的临时变量d对象,相当于Date d = d;后面的d是传进来的,也就是通过相同类型对象来创建对象,所以这里会调用拷贝构造函数。 } //主函数 int main(){ Date d; cout << "===== 1 =====" << endl; fa(d); cout << "===== 2 =====" << endl; return 0; }
运行结果:
日期类的构造函数 ===== 1 ===== 拷贝构造函数Date (Date& d3) //fa函数进栈 日期的析构函数 //弹栈 ===== 2 ===== 日期的析构函数 Program ended with exit code: 0
再写一个参数是引用类型的函数 fb 如下:
void fb(Date &d){ //这里是给传进的参数创建别名(引用)不会创建新的对象 }
修改主函数如下:
/主函数 int main(){ Date d; cout << "===== 1 =====" << endl; fa(d); cout << "===== 2 =====" << endl; fb(d); cout << "===== 3 =====" << endl; return 0; }
运行结果可以看出在2和3之间并没有创建新的对象
日期类的构造函数 ===== 1 ===== 拷贝构造函数Date (Date& d3) 日期的析构函数 ===== 2 ===== ===== 3 ===== 日期的析构函数 Program ended with exit code: 0
再创建一个参数类型是指针类型的函数 fc 如下:
void fc(Date *d){ //传进来的参数是指针类型 //这里创建了一个指针变量放在栈里面了,但是没有创建新的对象,指针指向的是传进来的对象的地址 }
主函数如下:
//主函数 int main(){ Date d; cout << "===== 1 =====" << endl; fa(d); cout << "===== 2 =====" << endl; fb(d); cout << "===== 3 =====" << endl; fc(&d); cout << "===== 4 =====" << endl; return 0; }
运行的结果如下:
日期类的构造函数 ===== 1 ===== 拷贝构造函数Date (Date& d3) 日期的析构函数 ===== 2 ===== ===== 3 ===== ===== 4 ===== 日期的析构函数从运行结果3和4之间也可以知道并没有创建新的对象。
再来,创建一个返回值类型是Date ,参数是引用类型的函数 fd 如下:
Date fd(Date &r){ return r; }
主函数中:
//主函数 int main(){ Date d; cout << "===== 1 =====" << endl; fa(d); cout << "===== 2 =====" << endl; fb(d); cout << "===== 3 =====" << endl; fc(&d); cout << "===== 4 =====" << endl; fd(d); cout << "===== 5 =====" << endl; return 0; }
运行结果如下:
日期类的构造函数 ===== 1 ===== 拷贝构造函数Date (Date& d3) 日期的析构函数 ===== 2 ===== ===== 3 ===== ===== 4 ===== 拷贝构造函数Date (Date& d3) 日期的析构函数 ===== 5 ===== 日期的析构函数 Program ended with exit code: 0
如果main主函数中修改如下,结果会如何呢?
cout << "===== 4 =====" << endl; Date d2 = fd(d); cout << "===== 5 =====" << endl;
理论上fd(d); 这里有一个拷贝构造函数和一个析构函数 还有这一步Date d2 = fd(d); 这里应该还有一个拷贝构造函数
运行结果如下:
日期类的构造函数 ===== 1 ===== 拷贝构造函数Date (Date& d3) 日期的析构函数 ===== 2 ===== ===== 3 ===== ===== 4 ===== 拷贝构造函数Date (Date& d3) ===== 5 ===== 日期的析构函数 日期的析构函数 Program ended with exit code: 0
为什么运行结果和我们分析的理论不一样呢??45之间为什么不是2个拷贝构造函数和1个析构函数呢?(其中一个拷贝构造函数和析构函数是fd的返回值临时变量的)
那是因为编译器做的优化,它把那个临时变量的拷贝构造函数和析构函数相抵消了,所以这里也就只显示了 Date d2 = ...这个的拷贝构造函数。
再写一个返回值是引用类型 参数是引用类型的函数 fe:
Date& fe(Date &r){ return r; }
主函数中添加如下代码:
<p class="p1"><span class="s1"> cout</span><span class="s2"> << </span><span class="s3">"===== 5 ====="</span><span class="s2"> << </span><span class="s4">endl</span><span class="s2">;</span></p><p class="p2"><span class="s5"> </span><span class="s6">fe</span><span class="s5">(d);</span><span class="s7">//</span><span class="s3">参数</span><span class="s7"> </span><span class="s3">返回值都是引用类型的函数</span></p><p class="p1"><span class="s2"> </span><span class="s1">cout</span><span class="s2"> << </span><span class="s3">"===== 6 ====="</span><span class="s2"> << </span><span class="s4">endl</span><span class="s2">;</span></p><p class="p3"><span class="s3"> </span><span class="s8">Date</span><span class="s3"> d3 = </span><span class="s9">fe</span><span class="s3">(d);</span></p><p class="p1"><span class="s2"> </span><span class="s1">cout</span><span class="s2"> << </span><span class="s3">"===== 7 ====="</span><span class="s2"> << </span><span class="s4">endl</span><span class="s2">;</span></p><p class="p1"><span class="s1"> Date</span><span class="s2"> &d4 = </span><span class="s3">fe</span><span class="s2">(d); //创建一个引用变量接收函数返回的</span></p><p class="p2"><span class="s4"> </span><span class="s5">cout</span><span class="s4"> << </span><span class="s2">"===== 8 ====="</span><span class="s4"> << </span><span class="s6">endl</span><span class="s4">;</span></p>
运行结果如下:
日期类的构造函数 ===== 1 ===== 拷贝构造函数Date (Date& d3) 日期的析构函数 ===== 2 ===== ===== 3 ===== ===== 4 ===== 拷贝构造函数Date (Date& d3) ===== 5 ===== ===== 6 ===== 拷贝构造函数Date (Date& d3) <p class="p1"><span class="s1">===== 7 =====</span></p><p class="p1"><span class="s1">===== 8 =====</span></p><p class="p2"><span class="s1">日期的析构函数</span></p><p class="p2"><span class="s1">日期的析构函数</span></p><p class="p2"><span class="s1">日期的析构函数</span></p><p class="p3"><span class="s1">Program ended with exit code: 0</span></p>
而在67之间因为通过返回值创建了对象 Date d3 =......这里调用了拷贝构造函数。
而78之间因为创建一个引用类型的变量来接收函数返回的,也就是相当于起别名,而并没有创建新对象,所以并没有调用拷贝构造函数
再来,在main函数当中添加如下代码:
cout << "===== 8 =====" << endl; Date d5[2] = {d,d2}; //数组里面有2个元素,分别用d和d2来赋值 <span class="s1" style="font-family: Arial, Helvetica, sans-serif;">cout</span><span class="s2" style="font-family: Arial, Helvetica, sans-serif;"> << </span><span class="s3" style="font-family: Arial, Helvetica, sans-serif;">"===== 9 ====="</span><span class="s2" style="font-family: Arial, Helvetica, sans-serif;"> << </span><span class="s4" style="font-family: Arial, Helvetica, sans-serif;">endl</span><span class="s2" style="font-family: Arial, Helvetica, sans-serif;">;</span><p class="p2"><span class="s3"> </span><span class="s5">Date</span><span class="s3"> d6[</span><span class="s6">5</span><span class="s3">];</span></p><p class="p1"><span class="s2"> </span><span class="s1">cout</span><span class="s2"> << </span><span class="s3">"===== 10 ====="</span><span class="s2"> << </span><span class="s4">endl</span><span class="s2">;</span></p>运行结果如下:
<p class="p1"><span class="s1"><span style="font-size:18px;">日期类的构造函数</span></span></p><p class="p2"><span class="s1"><span style="font-size:18px;">===== 1 =====</span></span></p><p class="p2"><span style="font-size:18px;"><span class="s2">拷贝构造函数</span><span class="s1">Date (Date& d3)</span></span></p><p class="p1"><span class="s1"><span style="font-size:18px;">日期的析构函数</span></span></p><p class="p2"><span class="s1"><span style="font-size:18px;">===== 2 =====</span></span></p><p class="p2"><span class="s1"><span style="font-size:18px;">===== 3 =====</span></span></p><p class="p2"><span class="s1"><span style="font-size:18px;">===== 4 =====</span></span></p><p class="p2"><span style="font-size:18px;"><span class="s2">拷贝构造函数</span><span class="s1">Date (Date& d3)</span></span></p><p class="p2"><span class="s1"><span style="font-size:18px;">===== 5 =====</span></span></p><p class="p2"><span class="s1"><span style="font-size:18px;">===== 6 =====</span></span></p><p class="p2"><span style="font-size:18px;"><span class="s2">拷贝构造函数</span><span class="s1">Date (Date& d3)</span></span></p><p class="p2"><span class="s1"><span style="font-size:18px;">===== 7 =====</span></span></p><p class="p2"><span class="s1"><span style="font-size:18px;">===== 8 =====</span></span></p><p class="p2"><span style="font-size:18px;"><span class="s2">拷贝构造函数</span><span class="s1">Date (Date& d3)</span></span></p><p class="p2"><span style="font-size:18px;"><span class="s2">拷贝构造函数</span><span class="s1">Date (Date& d3)</span></span></p><p class="p2"><span class="s1"><span style="font-size:18px;">===== 9 =====</span></span></p><p class="p1"><span class="s1"><span style="font-size:18px;">日期类的构造函数</span></span></p><p class="p1"><span class="s1"><span style="font-size:18px;">日期类的构造函数</span></span></p><p class="p1"><span class="s1"><span style="font-size:18px;">日期类的构造函数</span></span></p><p class="p1"><span class="s1"><span style="font-size:18px;">日期类的构造函数</span></span></p><p class="p1"><span class="s1"><span style="font-size:18px;">日期类的构造函数</span></span></p><p class="p2"><span class="s1"><span style="font-size:18px;">===== 10 =====</span></span></p><p class="p1"><span class="s1"><span style="font-size:18px;">日期的析构函数</span></span></p><p class="p1"><span class="s1"><span style="font-size:18px;">日期的析构函数</span></span></p><p class="p1"><span class="s1"><span style="font-size:18px;">日期的析构函数</span></span></p><p class="p1"><span class="s1"><span style="font-size:18px;">日期的析构函数</span></span></p><p class="p1"><span class="s1"><span style="font-size:18px;">日期的析构函数</span></span></p><p class="p1"><span class="s1"><span style="font-size:18px;">日期的析构函数</span></span></p><p class="p1"><span class="s1"><span style="font-size:18px;">日期的析构函数</span></span></p><p class="p1"><span class="s1"><span style="font-size:18px;">日期的析构函数</span></span></p><p class="p1"><span class="s1"><span style="font-size:18px;">日期的析构函数</span></span></p><p class="p1"><span class="s1"><span style="font-size:18px;">日期的析构函数</span></span></p><p class="p3"><span class="s1"><span style="font-size:18px;">Program ended with exit code: 0</span></span></p>
在结果89之间看到有2个拷贝构造函数。因为是通过同类型对象来创建对象的,所以会调用2个拷贝构造函数。
在9和10 之间会看到5个构造函数,这里不是拷贝构造而是直接创建对象,所以是构造函数。
如果再在main函数里面添加如下的代码:
<span style="font-size:24px;"> </span><span style="font-size:18px;"><span class="s1" style="font-family: Arial, Helvetica, sans-serif;">cout</span><span class="s2" style="font-family: Arial, Helvetica, sans-serif;"> << </span><span class="s3" style="font-family: Arial, Helvetica, sans-serif;">"===== 10 ====="</span><span class="s2" style="font-family: Arial, Helvetica, sans-serif;"> << </span><span class="s4" style="font-family: Arial, Helvetica, sans-serif;">endl</span><span class="s2" style="font-family: Arial, Helvetica, sans-serif;">;</span></span><p class="p2"><span style="font-size:18px;"><span class="s3"> d = d2; </span><span class="s5">//</span><span class="s6">赋值操作</span></span></p><p class="p1"><span style="font-size:18px;"><span class="s2"> </span><span class="s1">cout</span><span class="s2"> << </span><span class="s3">"===== 11 ====="</span><span class="s2"> << </span><span class="s4">endl</span><span class="s2">;</span></span></p><p class="p2"><span style="font-size:18px;"><span class="s3"> </span><span class="s7">Date</span><span class="s3"> *d7 = </span><span class="s8">new</span><span class="s3"> </span><span class="s7">Date</span><span class="s3">;</span></span></p><p class="p2"><span style="font-size:18px;"><span class="s3"></span></span></p><p class="p1"><span style="font-size:18px;"><span class="s1"> delete</span><span class="s2"> d7;</span></span></p><p></p><p class="p1"><span style="font-size:18px;"><span class="s2"> </span><span class="s1">cout</span><span class="s2"> << </span><span class="s3">"===== 12 ====="</span><span class="s2"> << </span><span class="s4">endl</span><span class="s2">;</span></span></p>运行结果:
<p class="p1"><span class="s1">===== 10 =====</span></p><p class="p1"><span class="s1">===== 11 =====</span></p><p class="p2"><span class="s1">日期类的构造函数</span></p><p class="p2"><span class="s1"></span></p><p class="p1"><span class="s1">日期的析构函数</span></p><p></p><p class="p1"><span class="s1">===== 12 =====</span></p>
而11和12之间只有一个构造函数,Date *d7是指针,指向一个对象,在创建这个对象的时候就调用了构造函数。
如果把11和12当中 delete d7注释掉 然后通过d7来创建对象,main函数中增加如下代码呢:
cout << "===== 12 =====" << endl; Date *d8 = new Date(*d7); delete d8; cout << "===== 13 =====" << endl;运行结果如下:
<p class="p1"><span class="s1">===== 12 =====</span></p><p class="p1"><span class="s2">拷贝构造函数</span><span class="s1">Date (Date& d3)</span></p><p class="p2"><span class="s1">日期的析构函数</span></p><p class="p1"><span class="s1">===== 13 =====</span></p>
在12和13之间是通过*d7得到同类型的对象然后创建对象,所以这里会调用拷贝构造函数。
要搞明白:
什么时候调用构造函数 ? 什么时候调用拷贝构造函数 ? 什么时候什么都不调用??
C++什么时候调用拷贝构造函数和什么时候调用构造函数
声明:以上内容来自用户投稿及互联网公开渠道收集整理发布,本网站不拥有所有权,未作人工编辑处理,也不承担相关法律责任,若内容有误或涉及侵权可进行投诉: 投诉/举报 工作人员会在5个工作日内联系你,一经查实,本站将立刻删除涉嫌侵权内容。