| 网站首页 | 业界新闻 | 小组 | 威客 | 人才 | 下载频道 | 博客 | 代码贴 | 在线编程 | 编程论坛
欢迎加入我们,一同切磋技术
用户名:   
 
密 码:  
共有 936 人关注过本帖
标题:析构函数
只看楼主 加入收藏
jfckpep
Rank: 1
等 级:新手上路
帖 子:40
专家分:2
注 册:2010-4-25
结帖率:100%
收藏
已结贴  问题点数:20 回复次数:14 
析构函数
#include<iostream>
using namespace std;

class MyClass
{
public:
    MyClass()
    {
        x=0;
        cout<<"default constructor of MyClass called!"<<endl;
    }
    MyClass(int i)
    {
        x=i;
        cout<<"Constructor of MyClass called! ("<<x<<")"<<endl;
    }
    ~MyClass()
    {
        cout<<"Destructor of MyClass called! ("<<x<<")"<<endl;
    }
    int GetX()
    {
        return x;
    }
    void Print()
    {
        cout<<x<<endl;
    }
private:
    int x;
};

class ExamClass
{
public:
    ExamClass()
    {
        val=0;
        cout<<"Default constructor of ExamClass called!"<<endl;
    }
    ExamClass(int i,int j,int k)/*:member_obj2(k),member_obj1(j)*/
    {
        member_obj2=k;
        member_obj1=j;//为什么将上面注释的部分改为这两行后,在程序没有结束就会调用析构函数呢?
        val=i;
        cout<<"constructor of ExamClass called!("<<val<<")"<<endl;
    }
    ~ExamClass()
    {
        cout<<"Destructor of ExamClass called!("<<val<<")"<<endl;
    }
    void Print()
    {
        cout<<val<<","<<member_obj1.GetX()<<","<<member_obj2.GetX()<<endl;
    }
private:
    MyClass member_obj1,member_obj2;
    int val;
};
int main()
{
    ExamClass obj1,obj2(1,2,3);
    system("pause");
    obj2.Print();
    system("pause");
    return 0;
}

问题为上面注释的中文说明处,还有麻烦说明下析构函数是何时被调用的?谢谢

[ 本帖最后由 jfckpep 于 2011-7-21 14:46 编辑 ]
搜索更多相关主题的帖子: void default private include 
2011-07-21 14:44
specilize
Rank: 4
等 级:业余侠客
帖 子:126
专家分:247
注 册:2011-2-20
收藏
得分:0 
ExamClass(int i,int j,int k)
    {
        member_obj2(k),member_obj1(j);
        val=i;
        cout<<"constructor of ExamClass called!("<<val<<")"<<endl;
    }
楼主你是说这样可以正确执行吗?不可能吧。
另外,在程序没有结束就会调用析构函数呢,具体什么意思啊

2011-07-21 19:52
jfckpep
Rank: 1
等 级:新手上路
帖 子:40
专家分:2
注 册:2010-4-25
收藏
得分:0 
回复 2楼 specilize
ExamClass(int i,int j,int k)/*:member_obj2(k),member_obj1(j)*/
     {
         member_obj2=k;
         member_obj1=j;//我的程序是这样的啊,不是“member_obj2(k),member_obj1(j);”
         val=i;
         cout<<"constructor of ExamClass called!("<<val<<")"<<endl;
     }
 
另外,运行程序,可以看到主函数还没结束时,就调用了析构函数~ExamClass,即可以看到输出了“Desturctor of MyClass called!"
2011-07-21 20:14
specilize
Rank: 4
等 级:业余侠客
帖 子:126
专家分:247
注 册:2011-2-20
收藏
得分:7 
回复 3楼 jfckpep
哦,如果我把程序改为这样
ExamClass(int i,int j,int k)
    {
        MyClass temp1(k),temp2(j);
        member_obj1=temp1;
        member_obj2=temp1;
        val=i;
        cout<<"constructor of ExamClass called!("<<val<<")"<<endl;
    }
那么毫无疑问,你知道肯定有Desturctor of MyClass called!这句话被输出两次,因为当temp1,temp2离开函数时,他会被析构
问题出在这
 member_obj2=k;    member_obj1=j
楼主你没发现这语法很奇怪吗,赋值操作符两边应该都是MyClass类型的,但是你把一个int赋给一个MyClass,却没有报错。在c++中,如果你没有为类写一个赋值操作符函数,会有一个合成的赋值操作符函数,但是编译器合成的赋值操作符函数接受的参数也是MyClass类型的,也就是k和j本应该为MyClass类型的,但是他不是,因为他不是,所以编译器会去寻找能否进行类型转换,使赋值操作成功,而刚好,你提供了这个函数 MyClass(int i)
这种叫做单参数的构造函数,当他没有被声明为explicit时,可以进行类型转换,也就是说,这里因为i为int,所以当需要一个MyClass时,int可以转换为MyClass,所以你的member_obj2=k;    member_obj1=j
这两句就是这种情况,所以编译器用k和j构造两个临时的MyClass变量,因为是临时的,所以当离开函数时,析构函数就调用了。楼主可以在MyClass(int i)前面加上explicit,这样编译器就会保错了
希望我说得够清楚
2011-07-21 21:26
jfckpep
Rank: 1
等 级:新手上路
帖 子:40
专家分:2
注 册:2010-4-25
收藏
得分:0 
回复 4楼 specilize
嗯,当时没考虑类型,谢谢了
你那个member_obj1=temp1;该将temp1改为temp2的
2011-07-21 21:42
zhangqi_gsts
Rank: 6Rank: 6
来 自:甘肃天水
等 级:侠之大者
威 望:1
帖 子:227
专家分:457
注 册:2011-3-27
收藏
得分:0 
菜鸟,飘过
2011-07-21 23:47
yuccn
Rank: 16Rank: 16Rank: 16Rank: 16
来 自:何方
等 级:版主
威 望:167
帖 子:6815
专家分:42393
注 册:2010-12-16
收藏
得分:0 
回复 4楼 specilize
specilize 功力深厚

我行我乐
公众号:逻辑客栈
我的博客:
https://blog.yuccn. net
2011-07-22 09:04
yuccn
Rank: 16Rank: 16Rank: 16Rank: 16
来 自:何方
等 级:版主
威 望:167
帖 子:6815
专家分:42393
注 册:2010-12-16
收藏
得分:7 
把它内存地址打印出来就可以看得很清楚了:
    MyClass()
    {
        x=0;
        cout<<"default constructor of MyClass called!"<<"  addr:"<<this<<endl;
    }
    MyClass(int i)
    {
        x=i;
        cout<<"Constructor of MyClass called! ("<<x<<")"<<"  addr:"<<this<<endl;
    }
    ~MyClass()
    {
        cout<<"Destructor of MyClass called! ("<<x<<")"<<"  addr:"<<this<<endl;
    }
图片附件: 游客没有浏览图片的权限,请 登录注册


在进入
ExamClass(int i,int j,int k)/*:member_obj2(k),member_obj1(j)*/
{
  //这里member_obj2 member_obj1已经调用了默认的构造函数了。
  member_obj2=k;// 这个时候。会有三步操作,1,调用MyClass(int i)构造一个临时对象,2再调用默认赋值运算符号赋值给member_obj2 ,
                 // 3 临时对象析构。 如果你从写一下赋值运算就看得很清楚了。
}

我行我乐
公众号:逻辑客栈
我的博客:
https://blog.yuccn. net
2011-07-22 09:23
jfckpep
Rank: 1
等 级:新手上路
帖 子:40
专家分:2
注 册:2010-4-25
收藏
得分:0 
回复 8楼 yuccn
那改成ExamClass(int i,int j,int k):member_obj2(k),member_obj1(j)的话就没有调用,两种方法有什么不一样吗?
2011-07-22 21:03
lintaoyn
Rank: 11Rank: 11Rank: 11Rank: 11
等 级:小飞侠
威 望:4
帖 子:606
专家分:2499
注 册:2009-4-8
收藏
得分:6 
我来说说吧
首先有两个名词,构造函数,和初始化列表(就是:member_obj2(k),member_obj1(j))具体内容和作用百度或谷歌
其次,类的数据成员的初始化不是在构造函数的函数体中进行的,是在初始化列表中进行的,初始化的顺序是按它们被声明顺序(可自己去查详细资料),如果你没有提供初始化列表的话,编译器会帮你搞定的,但前提是它能搞的定。
最后,在构造函数的函数体里进行的只会是数据成员的赋值。
程序代码:
class Test
{
public:
    Test(){}
    // Test():d(0){} 试着去编译这两种不同的构造函数
private:
    const int d;
};
int main()
{
    Test a;
    return 0;
}
让你困惑的应该不是析构函数在什么时候被调用。

迭代的是人,递归的是神。
2011-07-23 07:43
快速回复:析构函数
数据加载中...
 
   



关于我们 | 广告合作 | 编程中国 | 清除Cookies | TOP | 手机版

编程中国 版权所有,并保留所有权利。
Powered by Discuz, Processed in 0.036834 second(s), 8 queries.
Copyright©2004-2024, BCCN.NET, All Rights Reserved