再谈new和delete:
对于下面的这个例子:
String *favorite = new String(sayings);
其中,sayings为String类的对象,这个例子中,这不是为要存储的字符串分配内存,而是为对象分配内存;也就是说,为保存字符串地址的str指针和len成员分配内存。创建对象将调用构造函数,后者分配用于保存字符串的内存,并将字符串的地址赋给str。然后,当程序不再需要该对象时,使用delete删除它。这将只释放用于保存str指针和len成员的空间,并不释放str指向的内存,而该任务将由析构函数来完成。
在下述情况下析构函数将被调用:
- 如果对象是动态变量,则当执行完定义该对象的程序块时,将调用该对象的析构函数。
- 如果对象是静态变量,则在程序结束时将调用对象的析构函数。
- 如果对象是new创建的,则仅当您显式使用delete删除对象时,其析构函数才会被调用。
指针和对象小结:
使用对象指针时,需要注意几点:
- 使用常规表示法来声明指向对象的指针:
String * glamour;
- 可以将指针初始化为指向已有的对象:
String * first = sayings[0];
- 可以使用new来初始化指针,这将创建一个新的对象:
String *favorite = new String(sayings[choice]);
- 对类使用new将调用相应的类构造函数来初始化新创建的对象:
// invokes default constructor
String *gleep = new String;
// invokes the String(const char *) constructor
String *glop = new String("my my my");
// invokes the String(const String *) constructor
String * favorite = new String(sayings[choice]);
- 可以使用->运算符通过指针访问类方法:
if (sayings[i].length() < shortest->length())
- 可以对对象指针应用接触引用运算符(*)来获得对象:
if (sayings[i] < *first) first = &sayings[i];
再谈定位new运算符:
-
当运用定位new运算符时,程序员必须负责管用定位new运算符用从中使用的缓冲区内存单元。要使不同的内存单元,程序员需要提供两个位于缓冲区的不同地址,并确保这两个内存单元不重叠,例如:
pc1 = new (buffer)JustTesting; pc3 = new (buffer + sizeof(JustTesting))JustTesting("Better Idea", 6);
其中,pc1、pc3为类指针,buffer被定义为字符数组,JustTesting被定义为一个类,其中指针pc3相对于pc1的偏移量为JustTesting对象的大小。
-
如果使用定位new运算符来为对象分配内存,必须确保其析构函数被调用。delete可与常规new运算符配合使用,但不能与定位new运算符配合使用。因此,我们需要显式地为使用定位new运算符创建地对象调用析构函数。正常情况下将自动调用析构函数,这是需要显式调用析构函数地少数几种情形之一。
p3->~JustTesting(); p1->~JustTesting();
需要注意的一点是正确的删除顺序。对于使用定位new运算符创建的对象,应以与创建顺序相反的顺序进行删除。原因在于,晚创建的对象可能依赖于早创建的对象。另外,仅当所有对象都被销毁后,才能释放用于存储这些对象的缓冲区。
声明:以上整理自个人理解和Stephen Prata 著的《C++ Primer Plus》