C++_类和动态内存分配3-构造函数中使用new的注意事项

时间:2021-03-12 20:42:07

如果在构造函数中使用new来初始化对象的指针成员时必须特别小心。

1 如果在构造函数中使用new来初始化指针成员,则应在析构函数中使用delete。

2 new和delete必须相互兼容。new对应于delete,new[ ]对应于delete[ ]。

2      如果有多个构造函数,则必须以相同的方式使用new,要么都带中括号,要么都不带。因为只有一个析构函数,所有的构造函数都必须与它兼容。可以再一个构造函数中使用new初始化指针,也可以在另一个构造函数中使指针为空。

3      应定义一个复制构造函数,通过深度复制将一个对象初始化为另一个对象。通常,这种构造函数与下面类似。

String::String(const String & st)

{

num_string++;

len = st.len;

str = new char [len+1];

std::strcpy(str, st.str);

}

4      应当定义一个赋值运算符,通过深度复制将一个对象复制给另一个对象

 StringBad & StringBad::operator=(const StringBad & st)

{

       if(this == &st)

              return *this

       delete [] str;

       len = st.len;

       str = new char [len+1];

       std::strcpy(str, st.str);

       return *this;

}

 

总之该函数应当完成一些操作:检查自我赋值的情况,释放成员指针以前指向的内存,赋值数据而不仅仅是数据的地址,并返回一个指向调用对象的引用。

 

应该和不应该

 

包含类成员的类的逐成员复制

class Magazine

{

private:

       String title;

       string publisher;

};

String和string都使用动态内存分配,这是否意味着需要为Magazine类编写复制构造函数和赋值运算符?不,至少对这个类本身来说不需要。默认的逐成员复制和赋值行为有一定的智能。如果您将一个Magazine对象复制或赋值给另一个Magazine对象,逐成员复制将使用成员类型定义的赋值构造函数和赋值运算符。也就是说,复制成员title时,将使用String的复制构造函数,而将成员title赋给另一个Magazine对象时,将使用String赋值运算符,以此类推。

       然而,如果Magazine类因其他成员需要定义复制构造函数和赋值运算符,情况将更加复杂;在这种情况下,这些函数必须显式地调用String和string的复制构造函数和赋值运算符,这将在第13章中介绍。


显式调用。

隐式调用就是像正常使用内置类型那样初始化、赋值,符合传统习惯。