1.假设String类有如下私有成员:
class String
{
private:
char *str;
int len;
//...
};
请说出下面的构造函数有什么问题。
a.
String::String(){ }
b.
String::String(const char * s)
{
str = s;
len = strlen(s);
}
c.
String::String(const char * s)
{
strcpy(str,s);
len = strlen(s);
}
答:a.句法是正确的,但该构造函数没有将str指针初始化。该构造函数应将指针设置成NULL或使用new [ ]来初始化它。正确格式如下:
String::String()
{
str = NULL;
len = 0;
}
b.该构造函数没有创建新的字符串,而只是赋值了原有字符串的地址。它应当使用new [ ]和strcpy( )。
正确格式如下:
String::String(const char * s)
{
len = strlen(s);
str = new char [len+1];
strcpy(str,s);
}
c.它复制了字符串,但没有给它分配存储空间,应使用new char [len+1]来分配适当数量的内存。正确格式如 b 的代码。
.
.
2.如果你定义了一个类,其指针成员是使用 new 初始化的,请你指出可能出现的3个问题以及如何纠正这些问题。
答:首先,当这种类型的对象过期时,对象的成员指针指向的数据仍将保留在内存中,这将占用空间,同时不可访问,因为指针已经丢失。可以让类析构函数删除构造函数中 new 分配的内存,来解决这种问题。其次,析构函数释放这种内存后,如果程序将这样的对象初始化为另一个对象,则析构函数将试图释放这些内存两次,这是因为将一个对象初始化为另一个对象的默认初始化,将复制指针值,但不复制指向的数据,这将使两个指针指向相同的数据。解决方法是,定义一个赋值构造函数,是初始化复制指向的数据。第三,将一个对象赋给另一个对象也将导致两个指针指向相同的数据。解决方法是重载赋值操作符,使之赋值数据,而不是指针。
3.如果没有显示地提供类方法,编译器将自动生成哪些类方法?请描述这些隐式生成的函数的行为。
答:c++自动提供下面的成员函数:
- 默认构造函数
- 赋值构造函数
- 赋值构造函数
- 默认析构函数
- 地址操作符
默认构造函数不能完成任何工作,但使得能声明数组和未初始化的对象。
默认复制构造函数和默认赋值操作符使用成员赋值。
默认析构函数不完成任何工作。
隐式地址操作符返回调用对象的地址(即this指针的值)。