理解shared_ptr ---2

时间:2021-12-29 18:00:49

1、引用计数字段不能放在资源管理类中。我们的解决办法是,把引用计数和资源绑在一起,进行二次封装。但是这样存在一个大问题,不同类型的资源管理类不能兼容。也就是说,shared_ptr<Dog>不能赋值给shared_ptr<Animal>。

2、你可能会想,使用模版成员方法去解决,但是这里有个问题。因为进行了两次封装,u_ptr的类型还是不一样,也不能赋值。你可能会想,我在u_ptr中也建立模版成员方法,这也是错的。思考下,我们要保证,资源管理类指向同一个u_ptr,对u_ptr进行拷贝,那么资源管理类就不是指向同一个u_ptr了,这显然是错的。

3、有没有其它的办法呢?

  问题的关键是,进行了两次封装。不进行两次封装就好了,为了保证资源管理类指向同一个资源,同时指向同一个引用计数值,我可以再建立一个int指针,指向引用计数。

4、示例代码:

 template <typename T>
class shared_ptr_2
{
public:
shared_ptr_2(T* ptr):_ptr(ptr),_refCount_ptr(new int())
{
} shared_ptr_2(shared_ptr_2<T>& rhs)
{
_ptr= rhs._ptr;
_refCount_ptr = rhs._refCount_ptr;
++(*_refCount_ptr);
} template <typename U>
shared_ptr_2<T>(shared_ptr_2<U>& rhs)
{
_ptr = rhs._ptr;
_refCount_ptr = rhs._refCount_ptr;
++(*_refCount_ptr);
} shared_ptr_2& operator=(shared_ptr_2<T>& rhs)
{
if(this!=&rhs)
{
if(--(*_refCount_ptr) ==)
{
delete _ptr;
delete _refCount_ptr;
}
_ptr= rhs._ptr;
_refCount_ptr = rhs._refCount_ptr;
++(*_refCount_ptr);
}
return *this;
} template <typename U>
shared_ptr_2<T>& operator=(shared_ptr_2<U>& rhs)
{
++(*_refCount_ptr);
if(--(*_refCount_ptr) ==)
{
delete _ptr;
delete _refCount_ptr;
}
_ptr= rhs._ptr;
_refCount_ptr = rhs._refCount_ptr;
return *this;
} ~shared_ptr_2()
{
if(--(*_refCount_ptr) ==)
{
delete _ptr;
delete _refCount_ptr;
}
} private:
T* _ptr;
int* _refCount_ptr; friend class shared_ptr_2;
};