如果在构造函数中使用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章中介绍。
显式调用。
隐式调用就是像正常使用内置类型那样初始化、赋值,符合传统习惯。