波奇学C++:智能指针(二):auto_ptr, unique_ptr, shared_ptr,weak_ptr

C++98到C++11:智能指针分为auto_ptr, unique_ptr, shared_ptr,weak_ptr,这几种智能都是为了解决指针拷贝构造和赋值的问题

auto_ptr:允许拷贝,但只保留一个指向空间的指针。

管理权转移,把拷贝对象的资源管理权转移给拷贝对象,导致被拷贝对象悬空,不能访问出问题 ap1置空

auto_ptr ap1(new Test());
auto_ptr ap3 = ap1;
// ap1=nullptr

如果不置空就会导致对象调用析构函数时,空间会被释放两次。但会导致ap1无法再使用。

模拟auto_ptr

template
class AutoPtr
{
public:
	AutoPtr(T* ptr)
		:_ptr(ptr)
	{}
	T& operator*()
	{
		return *_ptr;
	}
	T* operator->()
	{
		return _ptr;
	}
	~AutoPtr()
	{
		cout << "delete pointer" << _ptr << endl;
		delete _ptr;
	}
	AutoPtr(AutoPtr& ap)
		:_ptr(ap._ptr)
	{
		ap._ptr = nullptr;
	}
private:
	T* _ptr;
};

 unique_ptr: 禁止拷贝

unique_ptr up1(new A());
unique_ptr up3=up1// 报错

模拟实现unique_ptr

template
class UniquePtr
{
public:
	UniquePtr(T* ptr)
		:_ptr(ptr)
	{}
	T& operator*()
	{
		return *_ptr;
	}
	T* operator->()
	{
		return _ptr;
	}
	~UniquePtr()
	{
		cout << "delete pointer" << _ptr << endl;
		delete _ptr;
	}
	UniquePtr(const UniquePtr& up) = delete; // 禁止拷贝构造
	UniquePtr& operator=(const UniquePtr& up) = delete; //禁止默认生成赋值重载函数
private:
	
	T* _ptr;
};

shared_ptr:支持拷贝,无法解决循环引用

shared_ptr通过引用计数来防止被多次析构

引用计数

波奇学C++:智能指针(二):auto_ptr, unique_ptr, shared_ptr,weak_ptr_第1张图片

用一个变量来记录指向空间的指针数,当一个智能指针释放时,变量数值减一,当数值为0时才释放空间。

模拟share_ptr的实现

template
class SharedPtr
{
public:
	SharedPtr(T* ptr)
		:_ptr(ptr)
		,_pcount(new int(1))
	{}
	T& operator*()
	{
		return *_ptr;
	}
	T* operator->()
	{
		return _ptr;
	}
	~SharedPtr()
	{
		if (--(*_pcount) == 0)
		{
			
			cout << "delete pointer" << _ptr << endl;
			delete _ptr;
			delete _pcount;
		}
	}
	
private:

	T* _ptr;
	int* _pcount;
};

 shared_ptr的拷贝构造

SharedPtr(const SharedPtr& sp)
		:_ptr(sp._ptr)
		,_pcount(sp._pcount)
	{
		(*_pcount)++;
	}
	

shared_ptr的赋值重载

SharedPtr& operator=(const SharedPtr& sp)
	{
		if (_ptr==sp._ptr)
		{
			return *this;
		}
		if (*_pcount == 1)
		{
			~SharedPtr();
		}
		else
		{
			(*_pcount)--;
			
		}
		*(sp._pcount)++;
		_pcount = sp._count;
		_ptr = sp._ptr;
		return *this;
	}

 循环引用问题

class Test
{
public:
	Test()
	{
		cout << "构造函数"<<" "< _next;
	shared_ptr _prev;
};
int main()
{
	shared_ptr sp1(new Node);
	shared_ptr sp2(new Node);
	sp1->_next = sp2;
    sp2->_prev=sp1
	return 0;
}

波奇学C++:智能指针(二):auto_ptr, unique_ptr, shared_ptr,weak_ptr_第2张图片

当sp1 和sp2析构时,引用计数减为1

波奇学C++:智能指针(二):auto_ptr, unique_ptr, shared_ptr,weak_ptr_第3张图片

当要释放左边空间->释放右边空间的_prev->释放左边的_next->释放左边的空间。

释放左边空间->释放左边空间

形成死循环无法解决。

weak_ptr:解决shared_ptr循环引用问题,不属于RAII智能指针

struct Node
{
	Test test;
	weak_ptr _next;
	weak_ptr _prev;
};

weak_ptr:解决原理不增加引用计数

template
class WeakPtr
{
public:
	WeakPtr()
		:_ptr(nullptr)
	{}
	WeakPtr(const SharedPtr& sp)
		:_ptr(sp._ptr)
	{}
	WeakPtr& operator= (SharedPtr& sp)
	{
		_ptr = sp.get();
		return *this;

	}
	T& operator*()
	{
		return *_ptr;
	}
	T* operator->()
	{
		return _ptr;
	}
private:
	T* _ptr;
};

你可能感兴趣的:(波奇学c,c++,开发语言)