1、智能指针(不带引用计数的智能指针)
1.1、基础知识
我们在使用智能指针之前,一直是使用的是裸指针。
问题:
- 如果我们不delete,会造成内存泄漏;
- 或者由于中间程序的执行顺序和我们预想的不一样,导致我们资源释放的代码写了,但是没有执行到,导致内存资源泄漏。
堆内存是我们用户手动开辟,手动释放
用不好裸指针也是造成我们内存资源非法访问,资源泄漏的一种因素
1.2、智能指针
裸指针: 用完没有delete,就会造成内存泄漏;
智能指针: 保证做到资源的自动释放;(利用栈上的对象出作用域自动析构的特征,来做到资源的自动释放)智能指针不能放在堆上。
智能指针就是对普通指针的一个封装;
智能指针能不能定义在堆上?
- 这样写在语法上没有问题,但是智能指针保证资源自动释放是在智能指针析构的时候。
- 但是现在这个智能指针被放在堆上,这个指针p虽然是智能指针类型,但是它前面是星号,就不是一个对象,它就是一个普通的裸指针,所以,要让堆上的这个智能指针析构,我们首先要delete p;
- 使用时,我们不能直接把智能指针定义在堆上,这样就没意义了。
#include <iostream>
using namespace std;
template<typename T>
class CSmartPtr
{
public:
CSmartPtr(T* p = nullptr) :ptr(p) {}
~CSmartPtr()
{
delete ptr;
}
T& operator*()
{
return *ptr;
}
T* operator->()
{
return ptr;
}
private:
T* ptr;
};
class Test
{
public:
void show()
{
cout << "liufeng" << endl;
}
};
int main()
{
CSmartPtr<int>p(new int);
*p = 20;
cout << *p << endl;
CSmartPtr<Test>q(new Test());
q->show();
(q.operator->())->show();
return 0;
}
1.3、不带引用计数的智能指针
智能指针使用时,包含#include <memory>
头文件 ;
不带引用计数的智能指针:
- auto_ptr:C++库里面
- scoped_ptr 和 unique_ptr : C++11新标准
1、auto_ptr
不带引用计数的智能指针怎么解决浅拷贝问题?
对于ptr2执行的拷贝构造函数:
- 将ptr2指针指向ptr1的内存空间,再将ptr 置为nullptr。
auto_ptr怎么解决浅拷贝问题?
- 在源码中,不带引用计数,成员变量只有一个裸指针;
- 怎么管理资源的?永远让最后一个智能指针管理资源,前面的智能指针全部被置为nullptr;
这也就造成上面程序的ptr1不能使用的情况;
不推荐使用auto_ptr;
容器中不能使用auto_ptr:
这种情况vec1中的auto_ptr会被全部置为nullptr;
2、scoped_ptr
-
比auto_ptr还暴力,使用的也比较少。
-
直接 将 类的拷贝构造 和 赋值重载delete了;
3、unique_ptr
-
不带引用技术的智能指针,推荐使用unique_ptr;
-
定义: 只让一个智能指针管理资源,不会让多个智能指针管理资源;
也 将 类的拷贝构造 和 赋值重载delete了:
这个拷贝构造时不能执行的:
解决方法:
- std::move(): 对对象的一个右值引用的强转。
将p1的资源移给p2,p1也是不可以再访问了;
- unique_ptr提供了右值引用的拷贝构造和赋值重载函数。
临时对象拷贝构造ptr1
unique_ptr的好处是用户自己知道用move把资源管理权移动了。