本文将简要介绍智能指针 shared_ptr 和 unique_ptr,并简单实现基于引用计数的智能指针。
使用智能指针的缘由
1. 考虑下边的简单代码:
1 | int main() |
就如上边程序,我们有可能一不小心就忘了释放掉已不再使用的内存,从而导致资源泄漏(resoure leak,在这里也就是内存泄漏)。
2. 考虑另一简单代码:
1 | int main() |
我们可能会心想,这下程序应该没问题了?可实际上程序还是有问题。上边程序虽然最后释放了申请的内存,但 ptr 会变成空悬指针(dangling pointer,也就是野指针)。空悬指针不同于空指针(nullptr),它会指向“垃圾”内存,给程序带去诸多隐患(如我们无法用if语句来判断野指针)。
上述程序在我们释放完内存后要将ptr置为空,即:
1 | ptr = nullptr; |
除了上边考虑到的两个问题,上边程序还存在另一问题:如果内存申请不成功,new 会抛出异常,而我们却什么都没有做!所以对这程序我们还得继续改进(也可用try…catch…):
1 |
|
3. 考虑最后一简单代码:
1 |
|
当我们的程序运行到 if(hasException()) 处且 hasException() 为真,那程序将会抛出一个异常,最终导致程序终止,而已申请的内存并没有释放掉。
当然,我们可以在 hasException() 为真时释放内存:
1 | // 假定 hasException 函数原型是 bool hasException() |
但,我们并不总会想到这么做。而且,这样子做也显得麻烦,不够人性化。
如果,我们使用智能指针,上边的问题我们都不用再考虑,因为它都已经帮我们考虑到了。
因此,我们使用智能指针的原因至少有以下三点:
- 智能指针能够帮助我们处理资源泄露问题;
- 它也能够帮我们处理空悬指针的问题;
- 它还能够帮我们处理比较隐晦的由
异常
造成的资源泄露。
智能指针
自 C++11 起,C++ 标准提供两大类型的智能指针:
Class shared_ptr 实现共享式拥有(shared ownership)概念。多个智能指针可以指向相同对象,该对象和其相关资源会在“最后一个引用(reference)被销毁”时候释放。为了在结构复杂的情境中执行上述工作,标准库提供了 weak_ptr、bad_weak_ptr 和 enable_shared_from_this 等辅助类。
Class unique_ptr 实现独占式拥有(exclusive ownership)或严格拥有(strict ownership)概念,保证同一时间内只有一个智能指针可以指向该对象。它对于避免资源泄露(resourece leak)——例如“以new 创建对象后因为发生异常而忘记调用 delete”——特别有用。
注:C++98 中的 Class auto_ptr 在 C++11 中已不再建议使用。
shared_ptr
几乎每一个有分量的程序都需要“在相同时间的多处地点处理或使用对象”的能力。为此,我们必须在程序的多个地点指向(refer to)同一对象。虽然 C++ 语言提供引用(reference)和指针(pointer),还是不够,因为我们往往必须确保当“指向对象”的最末一个引用被删除时该对象本身也被删除,毕竟对象被删除时析构函数可以要求某些操作,例如释放内存或归还资源等等。
所以我们需要“当对象再也不被使用时就被清理”的语义。Class shared_ptr 提供了这样的共享式拥有语义。也就是说,多个shared_ptr可以共享(或说拥有)同一对象。对象的最末一个拥有者有责任销毁对象,并清理与该对象相关的所有资源。
shared_ptr 的目标就是,在其所指向的对象不再被使用之后(而非之前),自动释放与对象相关的资源。
下边程序摘自《C++标准库(第二版)》 5.2.1 节:
1 |
|
程序运行结果如下:
关于程序逻辑可见下图:
关于程序的几点说明:
- 对智能指针 pNico 的拷贝是浅拷贝,所以当我们改变对象 Nico 的值为 Nicolai 时,指向它的指针都会指向新值。
- 指向对象 Jutta 的有四个指针:pJutta 和 pJutta 的三份被安插到容器内的拷贝,所以上述程序输出的 use_count 为 4。
- shared_ptr 本身提供默认内存释放器(default deleter),调用的是 delete,不过只对“由 new 建立起来的单一对象”起作用。当然我们也可以自己定义内存释放器,就如上述程序。值得注意的是,默认内存释放器并不能释放数组内存空间,而是要我们自己提供内存释放器,如:
1
2
3
4
5
6
7shared_ptr<int> pJutta2(new int[10],
// deleter (a lambda function)
[](int *p)
{
delete[] p;
}
);
或者使用为 unique_ptr 而提供的辅助函数作为内存释放器,其内调用 delete[]:
1 | shared_ptr<int> p(new int[10], default_delete<int[]>()); |
unique_ptr
unique_ptr 是 C++ 标准库自 C++11 起开始提供的类型。它是一种在异常发生时可帮助避免资源泄露的智能指针。一般而言,这个智能指针实现了独占式
拥有概念,意味着它可确保一个对象和其相应资源同一时间只被一个指针拥有。一旦拥有者被销毁或变成空,或开始拥有另一个对象,先前拥有的那个对象就会被销毁,其任何相应资源也会被释放。
现在,本文最开头的程序就可以写成这样啦:
1 |
|
智能指针简单实现
基于引用计数的智能指针可以简单实现如下(详细解释见程序中注释):
1 |
|
测试程序如下:
1 |
|
测试结果如下:
参考资料
- 《C++标准库(第二版)》
- C++中智能指针的设计和使用