为了更容易(同时也更安全)地使用动态内存,新的标准库提供了两种智能指针(smart pointer)类型来管理动态对象。智能指针的行为类似常规指针,重要的区别是它负责自动释放所指向的对象。新标准库提供的这两种智能指针的区别在于管理底层指针的方式:shared ptr
允许多个指针指向同一个对象; unique ptr
则“独占”所指向的对象。
上图中的get()
可以获取智能指针中保存的指针,需要注意不能get()
一个已经释放了的指针。
如:
std::shared_ptr<char> chptr = NULL;
chptr = std::shared_ptr<char>(new char[10]);
memset(chptr.get(), 0, 10);
char* pTemp = chptr.get();
2 shared_ptr的拷贝与赋值
我们可以认为每个shared_ptr
都有一个关联的计数器,通常称其为引用计数(reference count)。无论何时我们拷贝一个shared_ptr
,计数器都会递增。
一旦一个shared_ptr
的计数器变为0,它就会自动释放自己所管理的对象:
auto r =make_shared<int>(42) ;// r指向的int只有一个引用者
r =q; //给r赋值,令它指向另一个地址递增g指向的对象的引用计数1
//递减r原来指向的对象的引用计数
//r原来指向的对象已没有引用者,会自动释放
需要注意的是:拷贝一个shared_ptr
会递增其引用计数;将一个shared_ptr
赋予另一个shared_ptr
会递增赋值号右侧shared_ptr
的引用计数,而递减左侧shared_ptr
的引用计数。如果一个shared_ptr
的引用计数变为0,它所指向的对象会被自动销毁。
看下面几个例子:
// factory返回一个shared_ptr,指向一个动态分配的对象
shared ptr<Foo>factory(T arg)
{
//恰当地处理arg
// shared_ptr负责释放内存
return make shared<Foo>(arg);
}
void use_factory (T arg)shared_ptr<Foo>
{
p = factory (arg);
//使用p1
} //p离开了作用域,它指向的内存会被自动释放掉
void use_factory (T arg)shared_ptr<Foo>
{
p = factory (arg);
//使用p
return p; //当我们返回p时,引用计数进行了递增操作
} //p离开了作用域,但它指向的内存不会被释放掉
如果你将shared_ptr
存放于一个容器中,而后不再需要全部元素,而只使用其中一部分,要记得用erase
删除不再需要的那些元素。
默认情况下,std::shared_ptr
会调用delete
来清空内存。当使用new[]
分配内存时,需要调用delete[]
来释放内存,否则会有内存泄露。
可以通过以下代码来自定义释放内存的函数:
template< typename T >
struct array_deleter
{
void operator ()(T const * p)
{
delete[] p;
}
};
通过以下代码来声明std::shared_ptr
指针:
std::shared_ptr<int> sp(new int[10], array_deleter<int>());
此时,shared_ptr
可正确的调用delete[]
。
在C++11中,可以使用 std::default_delete
代替上面自己写的array_deleter
:
std::shared_ptr<int> sp(new int[10], std::default_delete<int[]>());
也可以使用一下的lambda
表达式来自定义删除函数
std::shared_ptr<int> sp(new int[10], [](int *p) { delete[] p; });
实际上,除非需要共享目标,否则unique_ptr
更适合使用数组:
std::unique_ptr<int[]> up(new int[10]); // this will correctly call delete[]
ps,上面代码可以正确的分配空间,但是空间内的值都没有初始化。如果需要默认初始化为0,可以使用下面的代码:
std::unique_ptr<int[]> up(new int[10]()); // this will correctly call delete[] 初始化为0
ps2,使用vector时,可以通过fill函数来将vector中所有元素置为默认值。
vector<unsigned char> data(dataLen);
std::fill(data.begin(), data.end(), 0);
参考博客:https://www.cnblogs.com/darkknightzh/p/5462363.html