C++副本构造器的问题
同样的一段代码在code::blocks运行没问题,但是在VC6.0中就出现问题;代码如下:int main()
{
MyClass obj1( new int(1) );
MyClass obj2( new int(2) );
obj2 = obj1;
obj1.print();
obj2.print();
std::cout << "----------------------\n";
MyClass obj3( new int(3) );
MyClass obj4 = obj3;
obj3.print();
obj4.print();
std::cout << "-----------------------\n";
MyClass obj5( new int(5) );
obj5 = obj5;
obj5.print();
return 0;
}
我猜:这个和副本构造器(拷贝构造函数没关系),是你没有重载operator=这个赋值运算符;这个赋值运算符的默认实现是浅拷贝,也就是说
obj2 = obj1;后,obj1和obj2中的int*指针指向了同一个new出来的对象(有没发现这里obj2new出来的new int(2)你已经无法访问了,这回导致内存泄漏),接着当main函数结束,obj2析构的时候delete一次该int*对象;obj2析构的时候,企图再次delete该int*所以报错。
code::blocks运行没问题不等于说它每次都运行没问题,每次都运行没问题也不等于说你的程序没有严重的问题。
最后,希望你不要总是把别人看成神仙。。。不把 MyClass的实现贴出来就问问题。。。 对,把myClass贴出来看看啊 #include <iostream>
#include <string>
class MyClass
{
public:
MyClass(int *p);
MyClass(const MyClass &rhs);
~MyClass();
MyClass &operator=(const MyClass &rhs);
void print();
private:
int *ptr;
};
MyClass::MyClass(int *p)
{
std::cout << "进入主构造器\n";
ptr = p;
std::cout << "离开主构造器\n";
}
MyClass::MyClass(const MyClass &rhs)
{
std::cout << "进入副本构造器\n";
*this = rhs;
std::cout << "离开副本构造器\n";
}
MyClass::~MyClass()
{
std::cout << "进入析构器\n";
delete ptr;
std::cout << "离开析构器\n";
}
//a = a;a = b;
//重载赋值操作符;
MyClass &MyClass::operator=(const MyClass &rhs)
{
std::cout << "进入赋值语句重载\n";
if( this != &rhs )
{
delete ptr;
ptr = new int;
*ptr = *rhs.ptr;
}
else
{
std::cout << "赋值号两边为同个对象,不作处理!\n";
}
std::cout << "离开赋值语句重载\n";
return *this;
}
void MyClass::print()
{
std::cout << *ptr << std::endl;
} 仰望天上的光 发表于 2015-4-9 21:55
我猜:这个和副本构造器(拷贝构造函数没关系),是你没有重载operator=这个赋值运算符;这个赋值运算符的 ...
我的错,没说清楚。在类里面重载了赋值运算符,就是main函数里,两个实例,如果去掉obj3和obj4,或者改成和obj1,obj2一样的赋值形式就没有问题。 xiaobei 发表于 2015-4-10 15:19
我的错,没说清楚。在类里面重载了赋值运算符,就是main函数里,两个实例,如果去掉obj3和obj4,或者改成 ...
引发问题的语句是:
MyClass obj4 = obj3;
这里调用了拷贝构造函数,然而你的拷贝构造函数直接调用了重载的赋值运算符。
重载的赋值运算符里做了:
if( this != &rhs )
{
delete ptr;
ptr = new int;
*ptr = *rhs.ptr;
}
这里delete ptr;会被执行。然而,你现在在构造函数中还没有为ptr new出一块空间,所以报错。 仰望天上的光 发表于 2015-4-10 15:48
引发问题的语句是:
MyClass obj4 = obj3;
这里调用了拷贝构造函数,然而你的拷贝构造函数直接调用了重 ...
但是那个代码的目的是为了测试直接定义个新MyClass变量,用boj3直接赋值给他的啊,目的不变的情况下怎么办。不然就直接MyClass obj4(new int (2));就达不到之前的目的了。 代码有问题 仰望天上的光 发表于 2015-4-10 15:48
引发问题的语句是:
MyClass obj4 = obj3;
这里调用了拷贝构造函数,然而你的拷贝构造函数直接调用了重 ...
此处如果注释掉delete ptr;应该会导致内存泄漏吧,那可不可以在前面加一句让*ptr=0;然后再删除呢,这样其实也起到了初始化指针指向内存的问题。
页:
[1]