Effective C++中文版学习记录(三)
章节三:资源管理
进度:17/55
文章目录
- [Effective C++中文版学习记录(三)](#Effective C++中文版学习记录(三))
- 条款13、以对象管理资源
- 条款14、在资源管理类中小心copying行为
- 条款15、在资源管理类中提供对原始资源的访问
- 条款16、成对使用new和delete时要采取相同形式
- 条款17、以独立语句将newed对象置入智能指针
条款13、以对象管理资源
考虑的是如果自己写指针,有以下情况:
cpp
void f
{
someClass * p = someClass();
...
delete p;
}
看起来很合理吧,申请资源,离开作用域时释放资源,防止内存泄漏
但是假设,这个代码需要进行迭代,多人合作,导致在"..."处出现了过早的return
那么p就不会被释放,造成内存泄漏
所以更合适的方式是都写进对象中,在离开作用域时让对象自己调用析构函数
或者使用shared_ptr和auto_ptr等等
PS.这里引入了C++中的一个概念,即RAII
条款14、在资源管理类中小心copying行为
资源管理类即条款13引入的RAII类,指的是资源在构造期间获得,在析构期间释放
这种RAII类的拷贝需要将它管理的资源一起拷贝,或者使用计数器,即shared_ptr等等
讲人话就是,不要只是复制指针,要一起复制资源,防止原本的类被释放了,拷贝的指针就会受影响
即
cpp
int * p1;
int * p2;
p2 = p1;
delete p1;
这种情况再使用p2就有问题了
条款15、在资源管理类中提供对原始资源的访问
简单而言就是,前文中描述RAII类能够自动管理这些资源,但是我们有时候需要直接使用这些资源,而不是通过RAII类
那么就需要RAII类来提供一个函数,能够直接返回这些资源,这个实现方法要看具体RAII类的设计方式
也就是类似于shared_ptr中的get函数,而返回资源可能需要经过一些转换,显式转换或隐式转换
尽量使用显式转换,因为比较安全
条款16、成对使用new和delete时要采取相同形式
举例
cpp
std::string * str = new std::string[100];
...
delete str;
这样是不对的,因为这样可能只是删除了string[0]这个部分,剩下的元素都没删除
正确做法应该是
cpp
delete [] str;
这样才能将完整的队列释放
条款17、以独立语句将newed对象置入智能指针
考虑这样一个情况
cpp
int num();
void test(shared_ptr<int> element, int num);
...
test(shared_ptr<int>(new int), num());
这里test接收两个元素,一个是shared_ptr指针,另一个是num()函数,而我们用一行处理可能会发生错误
如果这里num()函数调用错误,会导致shared_ptr指针里面new的元素丢失掉,无法释放,造成内存泄漏
所以应该拆开来写,即
cpp
shared_ptr<int> element(new int);
test(element, num());
也就是实现了独立语句创建智能指针,这样即便是num()函数崩了,element也还在,一样可以正常析构释放