上一节中讲了CFile文件操作类,本节主要来说说MFC异常处理。

在C++编程入门系列的最后一节:C++编程入门系列之五十(异常处理)中,讲了C++标准异常的处理机制,如果你还没有学过这方面内容,可以到那节教程中去学习下。

MFC异常处理与C++标准异常处理是类似的,只是它在抛出和捕获异常时使用了一些宏,另外还将异常封装到了CException类及其派生类中。下面就分别讲解这些宏与异常类。

MFC异常宏

MFC提供的异常处理宏包括TRY、CATCH、AND_CATCH、END_CATCH、THROW、THROW_LAST等,大家看着名称是不是与C++标准异常处理的关键字相似?它们实际上就是在try、catch和throw的基础上定义的。下面贴出MFC中这些宏的定义:

/   
// Exception macros using try, catch and throw   
//  (for backward compatibility to previous versions of MFC)   
   
#define TRY { AFX_EXCEPTION_LINK _afxExceptionLink; try {   
   
#define CATCH(class, e) } catch (class* e) \   
    { ASSERT(e->IsKindOf(RUNTIME_CLASS(class))); \   
        _afxExceptionLink.m_pException = e;   
   
#define AND_CATCH(class, e) } catch (class* e) \   
    { ASSERT(e->IsKindOf(RUNTIME_CLASS(class))); \   
        _afxExceptionLink.m_pException = e;   
   
#define END_CATCH } }   
   
#define THROW(e) throw e   
#define THROW_LAST() (AfxThrowLastCleanup(), throw)   
   
// Advanced macros for smaller code   
#define CATCH_ALL(e) } catch (CException* e) \   
    { { ASSERT(e->IsKindOf(RUNTIME_CLASS(CException))); \   
        _afxExceptionLink.m_pException = e;   
   
#define AND_CATCH_ALL(e) } catch (CException* e) \   
    { { ASSERT(e->IsKindOf(RUNTIME_CLASS(CException))); \   
        _afxExceptionLink.m_pException = e;   
   
#define END_CATCH_ALL } } }   
   
#define END_TRY } catch (CException* e) \   
    { ASSERT(e->IsKindOf(RUNTIME_CLASS(CException))); \   
        _afxExceptionLink.m_pException = e; } }
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.
  • 15.
  • 16.
  • 17.
  • 18.
  • 19.
  • 20.
  • 21.
  • 22.
  • 23.
  • 24.
  • 25.
  • 26.
  • 27.
  • 28.
  • 29.
  • 30.
  • 31.
  • 32.
  • 33.

可以看出这些宏的定义中都包含了相应的C++异常处理关键字,本质上还是要通过try、catch和throw实现。

MFC异常类

MFC将对异常的处理封装到了异常类–CException类及其子类中。其实即使我们不使用MFC异常宏而是使用C++标准异常处理的话,也是会用到MFC的CException类及其子类的。MFC异常类及其含义如下表:

MFC---异常处理(建议直接使用C++标准异常)(MFC常用类)_#define

MFC异常处理

MFC异常处理的TRY块的形式如下:

TRY  
{  
         复合语句  
}  
CATCH (MFC异常类名, 变量名)  
{  
         复合语句  
}  
AND_CATCH (MFC异常类名, 变量名)  
{  
         复合语句  
}  
AND_CATCH (MFC异常类名, 变量名)  
{  
         复合语句  
}  
......  
END_CATCH
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.
  • 15.
  • 16.
  • 17.
  • 18.

说明:TRY后的一对大括号内包含了可能会抛出异常的代码块;用CATCH子句捕获并处理异常,它捕获的是指向异常对象的指针,小括号中的“MFC异常类名”就是CException类或其子类的名称,变量名代表的就是“MFC异常类名”类型的指针变量;如果抛出的异常类型与CATCH子句中的不一致,则对后面的所有AND_CATCH子句依次检查,若子句的异常类型与抛出异常类型一致则由其捕获并处理此异常;最后用END_CATCH结束整个TRY块。

注意:MFC异常宏只能捕获处理CException及其子类类型的异常。

我们在使用MFC类时,有些会自动抛异常,当然我们可以在需要的情况下使用AfxThrow******()自己抛异常,这里的******与上面MFC异常类列表中的各个异常类对应,例如抛文件异常可以使用AfxThrowFileException(),参数可以查阅MSDN。

MFC异常处理实例

给大家一个简单的MFC异常处理的代码段,了解下如何使用MFC异常处理即可。

TRY   
{   
    CFile file(_T("C:\\1.txt"), CFile::modeRead);   // 构造CFile对象file,并以只读模式打开一个文件,如果不存在则抛出CFileException异常   
}   
CATCH (CFileException, e)   
{   
    if (e->m_cause == CFileException::fileNotFound)   
    {   
        // 如果捕获到CFileException异常且为文件未找到时,弹出提示对话框   
        MessageBox(_T("file not found!"));   
        return;   
    }   
}   
END_CATCH
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.

上面这段代码的意义很简单,就是打开一个文件C:\1.txt,如果此文件不存在,则抛出CFileException异常,由CATCH子句捕获后判断是否是文件未找到,如果是则弹出提示对话框并返回。

再将上面的代码稍微修改下,以演示AfxThrow******()抛异常的用法:

TRY   
{   
    AfxThrowFileException(CFileException::fileNotFound);   // 手动抛出CFileException异常   
}   
CATCH (CFileException, e)   
{   
    if (e->m_cause == CFileException::fileNotFound)   
    {   
        // 如果捕获到CFileException异常且为文件未找到时,弹出提示对话框   
        MessageBox(_T("file not found!"));   
        return TRUE;   
    }   
}   
END_CATCH
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.

上面这段代码执行时,在CATCH子句中会捕获到文件异常。
最后提醒大家一下,MFC的建议是不再使用MFC异常宏,而是直接使用C++标准异常,它更加灵活。