智能指针概述
动态内存
在C++中,动态内存的管理是用一对运算符完成的:new和delete,
- new:在动态内存中为对象分配一块空间并返回一个指向该对象的指针
- delete:指向一个动态对象的指针,销毁对象,并释放与之关联的内存。
动态内存管理经常会出现两种问题:
- 一种是忘记释放内存,会造成内存泄漏;
- 一种是尚有指针引用内存的情况下就释放了它,就会产生引用非法内存的指针。
如下代码,当一个操作发生异常时,可能导致delete操作不会执行,从而导致内存泄漏。而使用智能指针能轻易写出异常安全的代码,因为当对象退出作用域时,智能指针将自动调用对象的析构函数,避免内存泄露。
void func(){
auto ptr = new ClassA;
//执行一个会抛出异常的操作
func_throw_exception();
delete ptr;
}
使用了动态生存期的资源的类: 程序使用动态内存处于以下三种原因:
- 1、 程序不知道自己需要使用多少对象(容器类)
- 2、 程序不知道所需对象的准确类型
- 3、 程序需要在多个对象间共享数据(常见)
智能指针
为了更加容易(更加安全)的使用动态内存,引入了智能指针的概念。智能指针的行为类似常规指针,重要的区别是它负责自动释放所指向的对象。标准库提供的两种智能指针的区别在于管理底层指针的方法不同,shared_ptr允许多个指针指向同一个对象,unique_ptr则“独占”所指向的对象。标准库还定义了一种名为weak_ptr的伴随类,它是一种弱引用,指向shared_ptr所管理的对象,这三种智能指针都定义在memory头文件中。
智能指针是一个RAII类模型,用于动态分配内存,其设计思想是将基本类型指针封装为(模板)类对象指针,并在离开作用域时调用析构函数,使用delete删除指针所指向的内存空间。
作用:管理资源,避免内存泄漏。能够处理内存泄漏问题和空悬指针问题
智能指针shared_ptr
基本原理
C++ 11中最常用的智能指针类型为shared_ptr,允许多个指针指向同一对象。它采用引用计数的方法,记录当前内存资源被多少个智能指针引用。该引用计数的内存在堆上分配。当新增一个时引用计数加1,当过期时引用计数减一。只有引用计数为0时,智能指针才会自动释放引用的内存资源。
从而可以在任何地方都不使用时自动删除相关指针,从而帮助彻底消除内存泄漏和悬空指针的问题。每个 shared_ptr 对象在内部维护着两个内存位置:
1、指向对象的指针。
2、用于控制引用计数数据的指针。
template<class T>
class shared_ptr{
public:
...
private:
T* _ptr