目录
unique_ptr概述
常规初始方式
make_unique
unique_ptr常规操作
指定删除器
unique_ptr尺寸
unique_ptr概述
unique_ptr是一种独占式的智能指针,有着指向对象的专属所有权。相同时间段内,一个对象只能被一个shared_ptr对象指向。当unique_ptr离开作用域或者被重新赋值时,会自动删除所指向的对象,避免内存的泄露。unique_ptr可以移动,但不可以复制。不允许两个unique_ptr指向同一个对象。和shared_ptr一样,unique_ptr如果定义时不初始化,默认为空。
常规初始方式
unique_ptr<int> pu(new int(666));//定义指向值为666的int对象的unique_ptr智能指针
make_unique
和shared_ptr相似,weak_ptr也有与其相配对的初始化函数,就是make_ptr。使用make_unique初始化unique_ptr对象同样无法使用自定义的删除器。make_unique并不是在C++11中出现的,而是在C++14才被引入的。使用make_unique初始化unique_ptr对象性能更高,更安全。
unique_ptr<int> pu=make_unique<int>(666);//使用make_unique初始化
unique_ptr常规操作
unique_ptr不支持赋值和拷贝语义,但支持移动语义:
unique_ptr<string> pu(new string("Love mather"));
unique_ptr<string> pu2=std::move(pu);//pu置空,pu2指向原来的内存
release():放弃指向对象的专属所有权,返回指向指向对象的裸指针,并将该unique_ptr置空。返回的裸指针可以使用delete手动释放,也可以用来给其他智能指针初始化或赋值。如果使用release()一定要有相关的操作,无论是手工delete还是移交所属权,都要确保返回的裸指针指向的内存最终释放。
unique_ptr<int> pu(new int(111));
unique_ptr<int> pu2(pu.release());int *p=pu2.release();
delete p;
reset():释放该智能指针指向的对象,如果带参数,指向新的对象,如果不带参数,将该智能指针置空。
unique_ptr<int> pu1(new int(111));
unique_ptr<int> pu2(new int(222));
pu1.reset(pu2.release());//pu2置空,pu1指向对象内存释放,改为指向原pu2指向的内存。
pu1.release(new int(666));//释放原pu2指向的内存,pu1指向新的内存。
pu1.reset();//释放pu1指向的内存,并将pu1置空。
=nullptr:释放智能指针指向的对象,将智能指针置空
unique_ptr<int> pu(new int(666));
pu=nullptr;
指向数组:
unique_ptr<int []> pu(new int[10]);
pu[0]=0;
pu[1]=1;
pu[5]=2;
get():返回与智能指针相同指向的裸指针,同样不需要手动delete
unique_ptr<int> pu(new int(666));
int *p=pu.get();
*p=45;
swap():交换两个智能指针的指向对象
unique_ptr<int> pu(new int(666));
unique_ptr<int> pu2(new int(567));
std::swap(pu,pu2);//交换指向
pu.swap(pu2);//同样交换指向
转换为shared_ptr类型:可以将左值unique_ptr对象转换为shared_ptr
void func()
{return unique_ptr<int> pu(new int(345));
}shared_ptr<int> ps=func();
unique_ptr<int> pu2(new int(5667));
shared_ptr<int> ps2=std::move(pu2);
指定删除器
当系统提供的默认删除器不能满足需求时,我们也可以指定删除器进行一些附加操作。和shared_ptr相比,unique_ptr的删除器还需要在类型模板参数中添加删除器类型。曾经说过,shared_ptr即使删除器类型不同,只要指向对象相同,还是属于同一类型。但是unique_ptr不同。对于unique_ptr来讲,删除器的类型会影响对象的类型,两个不同删除器的unique_ptr对象类型不同,就不可以被放在同一容器中。相比之下,shared_ptr更灵活一点。
void mydelete(int *a)
{delete a;
}
typedef void(*p)(int *);//或者 using p = void(*)(int *);或者 typedef decltype(mydelete) *p;
unique_ptr<int,p> ps1(new int(666),mtdelete);
//使用lambda表达式
auto mydella = [](string *p){deldete p;
};
unique_ptr<string,decltype(mydella)> ps2(new string("无聊"),mydella);
unique_ptr尺寸
一般情况下,unique_ptr对象的尺寸和裸指针相同。当使用了自定义的删除器后,尺寸有可能会增加,也有可能不增加。具体来讲,lambda表达式作删除器并不会增加尺寸。使用自定义的函数作为删除器则会增加尺寸,对效率产生一定影响。但是shared_ptr的尺寸并不会受到删除器的影响,都是裸指针的两倍。