chiziting0315
2017-03-21 11:23
采纳率: 100%
浏览 859
已采纳

关于等号重载中构造函数中形参默认函数的问题,其中不懂为什么

#define _CRT_SECURE_NO_WARNINGS
#include

using namespace std;

//等号运算符的重载
class Array
{
public:
//int strlen(char *)
Array(char *t1=0)
{
this->length=strlen(t1);
this->p = new char [length +1];
strcpy(p,t1);
}
Array(const Array &t2)
{
this->length=t2.length;
this->p = new char [length +1];
strcpy(p,t2.p);
}

Array& operator=(Array &obj1)
{
    //先释放旧的内存
    if (this->p != NULL)
    {
        delete[] p;
        length = 0;
    }
    //2 根据obj1分配内存大小
    this->length = obj1.length;
    this->p = new char [length+1];

    //把obj1赋值
    strcpy(p, obj1.p);
    return *this;
}

~Array()
{
    if(*p !=NULL)
    {
        delete [] p;
        length=0;
    }
}
void printf()
{
    cout<<this->p;
}

private:
int length;
char *p;
protected:
};

int main()
{
Array a1("dasfsf");
Array a2=a1;//调用赋值构造函数,必须重新定义,要不然是浅拷贝,出错
a2.printf();
cout<<endl;
Array a4("dasda");
Array a3;
a3=a4;//浅拷贝,需重新定义等号。
a3.printf();
cout<<endl;

system("pause");
return 0;

}

源码如上,其中编译正确,却运行不了,如果把类中的构造函数的形参的默认赋值等于零,则正确,但是不知道为什么呢?

  • 写回答
  • 关注问题
  • 收藏
  • 邀请回答

4条回答 默认 最新

  • chiziting0315 2017-03-28 07:09
    已采纳

    在此感谢各位同僚的耐心解答,无奈同僚们回答的我都试过了,奖励金币不重要,学习到知识才是最重要的,因此,我自己苦心钻研,终于发现问题所在strlen()的参数不能是
    NULL,因此致错

    已采纳该答案
    打赏 评论
  • __fu 2017-03-21 13:09
    打赏 评论
  • 萌小宏 2017-03-22 04:42

    因为你定义a2调用无参的默认构造函数的时候并没有给它p分配空间,所以当你调用=运算符时一开始就释放空间,当然会出错,但当你在有一个形参那个构造函数中加=0时,那么这个一个形参的构造函数就会既接收无参的对象也接收一个形参的对象,这样,你再定义a2对象时就会调用带一个默认参数的构造函数,然后就会为p分配空间,然后你调用=运算符时释放时就有空间了,就不会出错了,希望我讲的对你有帮助!

    打赏 评论
  • myibu 2017-03-22 10:33

    #将 Array(char *t1=0)修改为 Array(char *t1="")

    Array a2=a1;//这里a2=a1相当于a2(a1),即通过调用Array类的拷贝构造函数完成a2对象的实例化
    Array a3;//通过调用Array类的默认参数构造函数完成对a3对象的实例化
    a3=a4;//对重载等号运算符的函数的调用完成对a3对象的赋值
    
    打赏 评论

相关推荐 更多相似问题