C++基本功和 Design Pattern系列 Exception

来源:互联网 发布:身份证注册软件下载 编辑:程序博客网 时间:2024/05/21 09:04

说到Exception就要说下相关的Error Handling. 比较常用的Error Handling一般有如下几种类方式:
    1. Return value
    2. Assert
    3. Debug Output
    4. Exception

相对于其他三种错误处理方式, Exception更加容易使用,而且使得错误代码相对集中,同时使得独立函数库的开发更加方便。同样,对于C++来说, Exception提供了Class的Constructor 和 Operator = 错误处理机制,因为这两者都不是能够通过return value进行报错的。

但是就游戏开发来说, Exception最大的缺点是内存和CPU的开销。当然,不是说游戏的代码中不应该使用Exception。 Aear见过用Exception的游戏代码,也有完全不用Exception的代码。因为对游戏来说,应该在运行过程中保持自身状态的正确性,不应该产生任何的无法处理的Exception。而所有能够自己处理的错误情况,都是能够通过Return value 来解决的。唯一可能产生Exception的地方,就是系统资源,比如磁盘文件,网络等。不过大部分系统掉用都提供非Exception的错误处理。不过程序开发各不相同,用不用Exception可能还是需要大家自行决定。

Aear个人观点是能不用Exception,就不用Exception,但是应该用Exception的时候,一定不要省。比如constructor里。

============ Exception的用法 ============

要使用Exception, 要么用系统的Exception的类,要么定义自己的类。在定义自己类的时候,可以继承STD里边的Exception类,也可以创建自己新的类。比如:

class ExceptionBase{
    ...
};

class ExceptionDerived : public ExceptionBase{
    ...
};

需要注意的是,通常定义自己的Exception类的时候,都要有一个公共的Base Exception Class, 这样能够保证写代码的时候catch所有的你自定义的Exception,比如:

try {
    ...
}catch( ExceptionDerived & e ) {
    ...
}catch( ExceptionBase & e ) {

    // Catch 其他的Exception, 这样的设计即使今后添加新的Exception,只要
    // 是从ExceptionBase继承来的,都会被catch到。

}catch( ... ) {

    // 这里最好再加上 catch(...)来catch所有的exception,防止有未catch的    // exception. 因为如果有unexpected exception, C++的缺省动作是直接
    // 终止程序的运行。

};

============ Exception in Constructor ============

如果一个Constructor产生exception而且没有被程序catch到的话,那么这个object的创建就会失败, 比如:

class MemoryBlock {
private:
    void * _pMem;
public:
    MemoryBlock ( UINT32 size )
    {
       _pMem = new char[size];
    };
    ....
};

MemoryBlock myMemory(100000000000000000000000000);

如果new在分配内存的过程中throw一个Exception ,通常是 bad_alloc,那么myMemory的创建就会失败,以后任何对 myMemory的成员访问,都是非法的,会导致程序的崩溃。

让我们看看另一中写法:

class MemoryBlock {
private:
    void * _pMem;
public:
    MemoryBlock ( UINT32 size ) :
       _pMem(new char[size])
    { };
    ....
};
上面也是合法的,不过会产生同样的问题。但是区别在于如果在代码中catch到exception,那么第一种写法,能够保证object被创建,而第二种写法不能。比如:

    // MemoryBlock 能够被创建
    MemoryBlock ( UINT32 size )
    {
       try {
           _pMem = new char[size];
       } catch(...) {}
    };

    // MemoryBlock 创建失败
    MemoryBlock ( UINT32 size )
    try
       : _pMem(new char[size])
    { } catch(...) {};


============ Exception in Destructor ============

其实对于Destructor来说就一句话,不能在Destructor中Throw Exception。原因很简单,因为通常Destructor要么在Delete Object中掉用,要么在已经Throw了Exception的时候,由系统掉用。如果在Throw Exception的情况下再Throw Exception的话,那么程序就会强制终止。

============ Exception in Operator ============

这个是比较麻烦的,通常的Exception的处理有好几个级别, Basic, Strong, Nofail.我们这里只说下Strong Exception Safety。 下面是个例子:

class X {
    ...
private:
    void * _pMem1;
    UINT32 _pMemSize1;
    void * _pMem2;
    UINT32 _pMemSize2;

public:
    X& operator = ( const X & xo )
    {
        if( _pMem1 ) delete _pMem1;
        if( _pMem2 ) delete _pMem2;
        
        _pMem1 = new char[xo._pMemSize1];
        _pMem2 = new char[xo._pMemSize1];
        ...
    };
};

这里如果 _pMem2 = new char[xo._pMemSize1]; Throw一个Exception,那么X只是被Copy了一半。状态是不完整的。但是原来在pMem1&2中的数据已经消失了。如果是Strong Exception Safety,那么要求如果throw excpetion,那么class的数据应该恢复在之前的状态,比如经典的exception safe operator = 如下:

    X& operator = ( const X & xo )
    {
        X temp(xo);
        swap( *this, temp );
        return *this;
    };

swap是交换*this 和 Temp的所有数据。通常我们能够保证这个过程没有任何exception的产生。因此即使 temp(xo) throw一个exception, 也不会影响当前类的任何状态变化。


============ RAII ============

最后说一种不使用Exception而能保证没有Resource Leakage的技术。那就是 Resource Aquisition Is Initialization ( RAII ). 其原理很简单,就是C++标准保证一个被成功创建的 Object,无论任何情况下(即使是在Throw exception ), 它的 Destructor都会被掉用。 因此,我们可以用一个object 的constructor 来获取资源,用Destructor来释放资源。下面举个最简单的应用,thread 的 asynchronization:

class CriticalSection {
public:
    CriticalSection( CRTICIAL_SECTION *pCs ) :
       _pCs(pCS)
    { EnterCriticalSection( _pCS ) };

    ~CriticalSection( )
    { LeaveCriticalSection( _pCS ) };
private:
    CRTICIAL_SECTION * _pCs;
};

通常我们使用Critical Section的时候,用下列方式:

void threadXX( CRTICIAL_SECTION * pCs)
{
    EnterCriticalSection( pCS );

    void * pTemp = new char[100000000];

    LeaveCriticalSection( pCS );
}

问题是如果     void * pTemp = new char[100000000]; Throw一个 bad_alloc,那么 LeaveCriticalSection( pCS );就不会被掉用而直接返回,很容易导致死锁。类似的代码在游戏服务器端的设计是很常见的,正确的做法是使用上面定义的类:

void threadXX( CRTICIAL_SECTION * pCs)
{
    CriticalSection temp( pCS );

    void * pTemp = new char[100000000];
}

由于即使throw exception, C++保证temp的destructor一定会被调用。因此不会产生死锁的情况。

============ 其他 ============

比如下面的代码是很容易产生问题的:
    function( new char[100], new char[300] );
如果new char[300]throw exception,那么 new char[100]很有可能就不会被释放。

推荐使用auto_ptr或者boost中的Shared_ptr,特别是在class 的initialization list 中, 比如下列做法不使用catch exception也不会产生内存泄露:

class X{
    X() :
    _ptr1(new XXX() ),
    _ptr2(new XXX() )
    {};

private:
    auto_ptr<void *> _ptr1;
    auto_ptr<void *> _ptr2;
}

Destructor中不需要catch exception,因为destructor主要是调用其他的destructor,没有任何的destructor会throw exception的,所以没必要catch.