免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
12下一页
最近访问板块 发新帖
查看: 4679 | 回复: 16
打印 上一主题 下一主题

[C++] 等号赋值的对象为什么销毁不调用析构函数呢? [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2013-05-07 14:58 |只看该作者 |正序浏览
Stock.cpp

  1. #include "Stock.h"
  2. #include <string>
  3. #include <iostream>

  4. using namespace std;

  5. Stock::Stock(string name, double price) {
  6.     cout << "with param constractor" << endl;
  7.     Stock::m_name = name;
  8.     Stock::m_price = price;

  9. }

  10. Stock::Stock(double price) {
  11.     Stock::m_name = "defautl name";
  12.     Stock::m_price = price;
  13.     cout << "with param constractor" << endl;
  14. }

  15. Stock::Stock() {
  16.     cout << "with no param constractor" << endl;
  17. }

  18. Stock::~Stock() {
  19.     cout << "destory stock name:" << Stock::m_name << " ByeBye!" << endl;
  20. }

复制代码

  1. Stock.h

  2. #ifndef STOCK_H_
  3. #define STOCK_H_

  4. #include <string>
  5. #include <iostream>

  6. using namespace std;

  7. class Stock {

  8. private:
  9.     string m_name;
  10.     double m_price;

  11. public:
  12.     Stock::Stock();
  13.     Stock::Stock(double price = 0.0);
  14.     Stock::Stock(string name = "defalut name", double price = 0.0);
  15.     virtual ~Stock();
  16. };

  17. #endif /* STOCK_H_ */
复制代码
main调用的如果是

  1. Stock myStock=2;
  2. myStock=Stock("a",1.0);
  3. myStock=Stock("abcde",3.0);
复制代码
则打印:
with param constractor
with param constractor
destory stock name:a ByeBye!
with param constractor
destory stock name:abcde ByeBye!
destory stock name:abcde ByeBye!

析构函数调用三次!
myStock = 2; 创建后,然后这个对象后来被销毁了,
因为myStock=Stock("a",1.0);
那为什么没有调用2这个对象的析构函数呢?


main调用的如果是

  1. Stock myStock=Stock("a",1.0);
  2. myStock=2;
  3. myStock=Stock("abcde",3.0);
复制代码
则打印:
with param constractor
destory stock name:a ByeBye!
with param constractor
destory stock name:defautl name ByeBye!
with param constractor
destory stock name:abcde ByeBye!
destory stock name:abcde ByeBye!

析构函数调用四次!这个2对象被销毁了。


那为什么第一个不销毁那个2对象呢?

论坛徽章:
1
2015年辞旧岁徽章
日期:2015-03-03 16:54:15
17 [报告]
发表于 2013-05-08 23:00 |只看该作者
constructor -- copy constructor -- move constructor
copy assignment -- move assignment

论坛徽章:
4
白羊座
日期:2013-09-17 21:59:30技术图书徽章
日期:2013-10-12 22:16:03白羊座
日期:2013-10-14 11:01:40双子座
日期:2013-12-17 18:26:39
16 [报告]
发表于 2013-05-08 21:33 |只看该作者
回复 14# ppm10103
你没有重载以int为参数的operator=函数,c++中只有一个参数的类成员函数也可以用来做隐式类型转换(带explicit关键字除外),
所以myStock=2实际的流程应当类似这样: 2 --> double(int) --> Stock(double) --> operator=(const Stock&);
如果你将Stock(double)构造函数带上explicit,你的程序将变异失败。
   

论坛徽章:
6
技术图书徽章
日期:2013-11-13 11:11:27子鼠
日期:2014-02-20 17:54:13处女座
日期:2014-06-16 17:43:33午马
日期:2014-08-08 09:11:17未羊
日期:2014-08-10 11:57:072015年辞旧岁徽章
日期:2015-03-03 16:54:15
15 [报告]
发表于 2013-05-08 15:56 |只看该作者
本帖最后由 littledick 于 2013-05-08 15:57 编辑

拷贝构造和赋值操作都没搞清楚么{:3_185:}
还滥用默认参数。

论坛徽章:
0
14 [报告]
发表于 2013-05-08 13:45 |只看该作者
回复 9# learningl


是VC++2010

Stock myStock = Stock("a", 1.0);
myStock = 2;

with param constractor
destory stock name:a ByeBye! 临时1.0对象析构
with param constractor
destory stock name:defautl name ByeBye! 临时2对象析构
destory stock name:defautl name ByeBye! myStock对象析构


那这个myStock = 2应该也不会生成临时变量才对啊

   

论坛徽章:
0
13 [报告]
发表于 2013-05-08 13:35 |只看该作者
本帖最后由 ppm10103 于 2013-05-08 13:36 编辑

多谢各位了,拷贝构造的

论坛徽章:
0
12 [报告]
发表于 2013-05-08 09:05 |只看该作者
井蛙夏虫 发表于 2013-05-07 20:23
2对象的name被myStock=Stock("a",1.0)改为a了,又被myStock=Stock("abcde",3.0)改为abcde。
它的析构对应最 ...


一个是三次构造函数调用,四次析构函数调用

一个是三次构造函数调用,三次析构函数调用

论坛徽章:
4
白羊座
日期:2013-09-17 21:59:30技术图书徽章
日期:2013-10-12 22:16:03白羊座
日期:2013-10-14 11:01:40双子座
日期:2013-12-17 18:26:39
11 [报告]
发表于 2013-05-07 20:23 |只看该作者
本帖最后由 井蛙夏虫 于 2013-05-07 20:26 编辑

2对象的name被myStock=Stock("a",1.0)改为a了,又被myStock=Stock("abcde",3.0)改为abcde。
它的析构对应最后一句destory stock name:abcde ByeBye!。

不过LZ的函数重载有问题。当这样定义(Stock myStock时,三个构造函数都匹配。

论坛徽章:
4
白羊座
日期:2013-09-17 21:59:30技术图书徽章
日期:2013-10-12 22:16:03白羊座
日期:2013-10-14 11:01:40双子座
日期:2013-12-17 18:26:39
10 [报告]
发表于 2013-05-07 20:19 |只看该作者
回复 9# learningl
LZ的结果是对的。第二个测试的myStock是调用拷贝构造函数创建的。



   

论坛徽章:
0
9 [报告]
发表于 2013-05-07 18:24 |只看该作者
LZ,在你说结果的时候,应该附上你所使用的编译器。

我的分析如下
----------------------------------------------------------
main调用的如果是
1    Stock myStock=2;
2   myStock=Stock("a",1.0);
3   myStock=Stock("abcde",3.0);

with param constractor                     -> 由1产生,此处并未产生临时变量。
with param constractor                     -> 由2Stock("a",1.0)产生,由于这里Stock("a",1.0),是一个临时变量,所以会有一个对应的析构
destory stock name:a ByeBye!          -> 2处对应的析构
with param constractor                     -> 同上面的道理,由3Stock("abcde",3.0)产生,同理是一个临时变量,所以会有一个析构
destory stock name:abcde ByeBye!  -> 3处对应的析构
destory stock name:abcde ByeBye!  -> 由1处产生,因为程序结束,所以myStock会调用析构,并且此时myStock中的m_name,已经被第3行的调用修改,所以也显示为abcde

----------------------------------------------------------
main调用的如果是
1    Stock myStock=Stock("a",1.0);
2    myStock=2;
3    myStock=Stock("abcde",3.0);

with param constractor
destory stock name:a ByeBye!
with param constractor
destory stock name:defautl name ByeBye!
with param constractor
destory stock name:abcde ByeBye!
destory stock name:abcde ByeBye!

析构函数调用四次!这个2对象被销毁了。
那为什么第一个不销毁那个2对象呢?

这个应该是错误的,构造的次数应该和析构的次数是成对的
我的结果如下 (g++ (GCC) 4.5.2)
with param constractor                                             
with param constractor
destory stock name:defautl name ByeBye!
with param constractor
destory stock name:abcde ByeBye!
destory stock name:abcde ByeBye!
原因同上面的分析。
  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP