std::shared_ptr及std::unique_ptr使用数组注意事项

1 智能指针类

为了更容易(同时也更安全)地使用动态内存,新的标准库提供了两种智能指针(smart pointer)类型来管理动态对象。智能指针的行为类似常规指针,重要的区别是它负责自动释放所指向的对象。新标准库提供的这两种智能指针的区别在于管理底层指针的方式:shared ptr允许多个指针指向同一个对象; unique ptr则“独占”所指向的对象。
std::shared_ptr及std::unique_ptr使用数组注意事项_第1张图片
上图中的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

你可能感兴趣的:(C++)