一般的智能指针都是通过一个普通指针来初始化,所以很容易写出以下的代码:
#include <iostream>
using
namespace std;
int
func1(){
//返回一个整数的函数
}
void
func2(AutoPtr<int*> ptr,int t){
//一些操作
}
int
main(){
func2(AutoPtr<int*>(new
int(5)),func1());
//其他操作
}
乍一看,这段代码好像没有什么问题,但实则暗藏隐患。
我们在调用func2的时候,里面的参数是需要运算的,运算一共有三步:
- int *t=new int(5);// 假设中间变量为t
- AutoPtr<int*> param(t);// 假设中间变量为param
- func1();
我们知道,1一定在2前面调用,但是3的调用顺序就不一定了.
我们假设一种情况:调用顺序为1->3->2
假设我们在调用3的时候,抛出了一个异常,这时候1已经调用完毕,也就是说,内存已经分配了,但是没有成功的放入智能指针中。这块内存不会被自动释放。于是引发了内存泄漏。
为了避免这个情况,我们可以将该过程分为两步:
#include <iostream>
using
namespace std;
int
func1(){
//返回一个整数的函数
}
void
func2(AutoPtr<int*> ptr,int t){
//一些操作
}
int
main(){
AutoPtr<int*>
param(new
int(5));
func2(param,func1());
//其他操作
}
只要确保智能指针的初始化不被打断,该隐患自然就会解除。
即使func1抛出异常,此时,智能指针已经初始化完毕,可以在析构函数中释放掉管理的内存。