条款16:成对使用new和delete时要采取相同形式
本条款内容比较简单,但却是经常遇到的错误操作。
看一下几种误操作
int *arr_i = new int[10];
delete arr_i;
string *arr_s = new string[10];
delete arr_s;
string *str2 = new string;
delete[] str2;
解析string类类型的操作过程,执行new操作,系统为其分配一块string大小的内存或者n块string大小的内存并加上n大小占用的内存,然后执行一个或多个string类构造函数(数组),delete操作时,先执行一次或多次析构函数,再释放对应的内存。
他们的内存分配情况如下(不同编译器可能不同)
只为arr_s执行delete操作,编译器只会执行一次析构函数,然后再释放内存,则程序会出现结果未定义,如果为str2执行delete[],编译器先在内存中找到数组大小n,然后根据大小执行n次析构函数,结果也是未定义 即使对于内置类型的int也是如此.
Typedef
typedef std::string stringArr[10];
std::string arr_string = new stringArr;
delete arr_string; //结果未定义
delete[] arr_string; //ok
为了避免这种情况,尽量不要使用typedef定义数组别名,可以通过string和vector替代。如上面的可以定义vector<std::string> arr_string
记住
如果你在new表达式中使用[],必须在相应的delete表达式中也使用[].如果你在new表达式中不使用[],一定不要在相应的delete表达式中使用[].