内存管理

1)正确得到: 正确调用内存分配和释放程序;

2)有效使用: 写特定版本的内存分配和释放程序;

C中用mallco分配的内存没有用free返回, 就会产生内存泄漏, C++中则是new和delete;

new和delete是隐式地调用构造和析构函数的, 而且可以在类内和类外自定义new和delete操作符, 这样带来了复杂性; 


条款5 对应的new和delete要采用相同的形式

 

1
2
3
string *stringArray =  new  string[100];
//...
delete  stringArray;

>stringArray指向的100个string对象中的99个不会被正确地摧毁, 析构函数不会被调用;

new会做两件事: 1) 内存被分配(operator new函数); 2) 为被分配的内存调用一个或多个构造函数;
delete也做两件事: 1) 为将被释放的内存调用一个或多个析构函数; 2) 释放内存(operator delete函数); 

删除数组时要使用delete[]:

 

1
2
3
4
5
string *stringPtr1 =  new  string;
string *stringPtr2 =  new  string[100];
//...
delete  stringPtr1;  // 删除一个对象
delete  [] stringPtr2;  // 删除对象数组

在delete非数组时加 [], 结果是不可预测的, 在delete数组时没有 [], 结果也是不可预测; 对于int这样的固定类型, 没有析构函数, 但结果也是不可预测;

Note new对应delete, new[]对应delete[]; 

当一个类包含指针数据成员, 有多个构造函数时尤其注意, 在所有初始化指针成员的构造函数里采用相同的new的形式, 否则在析构函数中无法确定delet的形式;

Note 对typedef来说也需注意, 为避免混乱, 最好杜绝对数组类型使用typedef; 使用stl的string和vector模板替代;

1
2
3
4
5
typedef  string AddressLines[4];
string *pal =  new  AddressLines;
// 注意"new AddressLines" 返回string*, 和 "new string[4]"返回的一样
delete  pal;  // 错误!
delete  [] pal;  // 正确

>AddressLines[]->vector<string>;


条款6 析构函数里对指针成员调用delete

大多数情况下, 动态内存分配的类在构造函数里用new分配内存, 在析构函数里用delete释放内存;

1) 每个构造函数里对指针进行初始化, 如果暂时没有内存分配给指针的话, 指针要被初始化为0(NULL, 空指针);
2) 删除现有的内存, 通过赋值操作符分配给指针新内存;
3) 析构函数里删除指针;

在构造和赋值操作过程中出现问题会比较明显, 但是在析构函数里没有正确删除指针, 可能只表现为一点微小的内存泄漏, 不断增长, 最后导致程序crash;

Note 删除空指针是安全的(等于什么都没做); 

在写构造函数, 赋值操作符, 或其他成员函数时, 类的指针成员要指向有效的内存或者指向为空, 这样在析构函数中可以简单地delete, 不用检查指针是否new过;

Note 对于没有用new初始化的指针, 就像你永远不会去删除传递进来的指针 [异步的情况下, 可能需要接受外部传递指针的控制权, 在内部删除外部临时new的指针]

使用智能指针可以避免必须删除成员指针, 把成员指针用智能指针对象代替; e.g. stl中的auto_ptr;

 

 

Copyright © 2024 冰天雪域
Powered by .NET 9.0 on Kubernetes