Effective C++中文版学习记录(三)

Effective C++中文版学习记录(三)

章节三:资源管理

进度:17/55


文章目录


条款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也还在,一样可以正常析构释放

相关推荐
Lei_3359671 分钟前
[数据结构]哈希表、字符串哈希
数据结构·哈希算法·散列表
微露清风1 小时前
系统性学习C++-第八讲-vector类
java·c++·学习
weixin_445251831 小时前
7sch C++ <B> weak_ptr circular reference 1/99
c++
给大佬递杯卡布奇诺2 小时前
FFmpeg 基本数据结构 AVInputFormat 分析
数据结构·c++·ffmpeg·音视频
月临水2 小时前
Git 学习笔记
笔记·git·学习·1024程序员节
奋斗的牛马2 小时前
FPGA—ZYNQ学习Debug(三)
学习·fpga开发
给大佬递杯卡布奇诺2 小时前
FFmpeg 基本数据结构 AVCodecContext分析
数据结构·c++·ffmpeg·音视频
qq_401700412 小时前
matlab学习
学习·算法·matlab
hd51cc2 小时前
C++ 类的学习(四) 继承
开发语言·c++·学习
紫荆鱼2 小时前
设计模式-命令模式(Command)
c++·后端·设计模式·命令模式