条款13:以对象管理资源
void f(){
InvestMent* pInv = createInvestment();
...
delete pInv;
}
对于上述情况,可能存在很多时候无法调用delete进行资源的释放。
1、过早的遇到一个return语句;
2、delete在循环中提前遇到break或continue;
3、遇到抛出异常的情况。
因此为了确保createInvestment返回的资源总是被释放,需要将资源放进对象内,当控制流离开函数f,该对象的析构函数会自动释放那些资源。
例如可以使用auto_ptr指针,其析构函数自动对其所指对象调用delete:
void f(){
std::auto_ptr<Investment> pInv(createInvestment());
...
}
auto_ptr定义的对象,若通过copy构造函数或copy assignment操作符来复制他们,他们会变成null,而复制所得的指针将取得资源的唯一拥有权。因此auto_ptr无法完成正常的赋值与复制操作。
解决方案是使用shared_ptr指针,使用方法类似,但是shared_ptr存在一个计数器用于追踪共有多少对象执行资源,并在无人指向它时自动删除该资源。
但是shared_ptr和auto_ptr是 使用delete进行释放,不是delete[],因此不要在动态分配得到的array身上使用shared_ptr和auto_ptr,如:
std::auto_ptr<std::string> aps(new string[10]);
std::shared_ptr<int> spi(new int[10]);
条款14:在资源管理类中小心copying行为
当一个对象被复制时,处理方式:
1、禁止复制。即吧复制构造函数及copying assignment设置为private。
2、对底层资源祭出“引用计数法”,通过只要内含一个shared_ptr成员变量,便可实现出引用计数法。甚至可以自定义删除器,而不是使用delete
class Lock{
public:
explicit Lock(Mutex* pm): mutexPtr(pm, unlock){ //指定unlock为删除器
lock(mutexPtr.get());
}
private:
std::str1::shared_ptr<Mutex> mutexPtr;
};
mutexPtr的析构函数会在互斥器的引用次数为0时自动调用删除器unlock
3、复制底层资源。即进行深度拷贝
4、转移底部资源的拥有权。即使用auto_ptr
条款15:在资源管理类中提供对原始资源的访问
shared_ptr 和 auto_ptr都提供了get函数来获得原始指针,也都对*和->进行了操作符重载
对于获得底部资源,可以提供两种方法:
1、使用get()函数获得对象的底部资源。
2、使用隐式转换函数
class Font{
FontHandle f;
public:
explicit Font(FontHandle fh):f(fh){}
operator FontHandle()const{ return f;}
}
int main(){
Font f(getFont());
changeFontSize(f); //将Font隐式转换为FontHandle
}
条款16:成对使用new和delete时要采用相同形式
new和delete以及new []和delete[]是配对的。
条款17:以独立语句将newed对象置于智能指针
shared_ptr的构造函数是explicit声明的,不能直接使用隐式装换,但可以强制转换,如
int priority();
void processWidget(shared_ptr<Widget> pw, int priority);
//在调用processWidget时有:
processWidget( shared_ptr<Widget>(new Widget()), priority)
上述processWidget调用方式可能造成资源泄露,C++没有明确参数核算的特定顺序,因此上述调用可能分为以下三部分组成:
1、执行new Widget
2、调用priority
3、调用shared_ptr
但是在调用priority时可能出现异常,导致new Widget无法使用shared_ptr进行释放,因此解决方案就是以独立语句将newed对象置入智能指针
即:
int priority();
void processWidget(shared_ptr<Widget> pw, int priority);
//在调用processWidget时有:
shared_ptr<Widget> pw(new Widget());
processWidget( pw, priority);