c++内存管理方式
概述
本章总结一些关于个人对内存管理的理解,主要包括如下内容:
内存管理原则
学c++的同学都知道这个内存管理原则,就是“谁创建,谁释放”或者说“谁申请,谁释放”。简单地说,在代码上体现为,调用new或malloc等内存分配的人,同时需在内存使用完成后调用delete或free释放。
这个原则看似大家都同意,毕竟只有申请内存的人,才知道什么时候该释放内存。
这无疑是一个正确的原则,但这个原则在任意场景中是否都合理呢,不见得。按照这个原则,作为一个接口提供者,要保证自己没有内存问题,其实很简单,接口提供者只需把内存申请和释放的工作都交给接口使用者就可以了。这看似很完美,我不创建,就不需要释放,也就不存在内存漏洞的问题,如果系统有内存泄露的问题,那肯定是接口使用者的责任。不知道真相的使用者只能“满眼泪光”地回去调逻辑,知道真相的使用者也只能无辜地回去调代码。
优秀的接口
接口设计者做对了,但不一定是做好了,或者说,接口设计者不一定做得够好了。
在我们评论接口设计者做得是否“够好了”之前,我们首先得确立一个简单评价的标准。
按照个人观察(并非独创想法),通俗总结,一个好的接口设计有如下特点:
- 满足使用者功能要求(当然是基础);
- 接口友好,容易理解和使用;
- 容易被正确使用(第二点的延伸);
- ……
按照以上标准看,作为接口使用者的我们来看,接口设计者似乎的确做得不够好,因为他总是把容易出错的部分交给接口使用者来完善,增加了接口使用者设计上的难度。不满足以上标准的第2、3点。
智能指针的作用在哪里?
OK,我们终于涉及智能指针了,在这里我们不打算讲智能指针的实现原理,这部分网上有很多可以参考的文章,另外,这部分也不需要大家去实现,现在c++11已经有标准的智能指针形式std:share_ptr,这意味着你想使用智能指针,是一件非常容易的事情。
使用智能指针,能让接口设计者做得更好:
智能指针的作用在于能自动释放指针,减少人为使用new和delete所存在的内存问题。接口设计者可以运用智能指针,把申请内存的工作都在接口内部实现并加以限制,把释放内存的工作交给智能指针。
常见的形式如下代码,接口提供者提供私有构造函数类,禁止外部直接创建,而接口使用者可以通过使用类的CreateInstance获得一个Test类对象。
这样,接口使用者不需要在关注内存释放问题。根据如下代码的形式,我们总结下只能指针的优点:
- 由于CreateInstance返回一个Test类型的智能指针对象,由智能指针负责释放。
- 接口使用者没有直接调用new申请内存故也无需调用delete释放。
- 接口设计设把类构造函数设为私有(我们在此不考虑这样做的拓展性方面的影响),限制外面使用new创建,接口友好度提高,而且更加容易被正确使用。
class Test { public: typedef std::shared_ptr<Test> TestPtr static TestPtr CreateInstance() { return new Test(); } private: Test() { }
};