抛出一个catch省略号(...)重新抛出C ++中的原始错误?

时间:2010-03-19 01:22:44

标签: c++ exception throw ellipsis

如果在我的代码中我有以下代码段:

try {
  doSomething();
} catch (...) {
  doSomethingElse();
  throw;
}

throw会重新抛出默认省略号处理程序捕获的特定异常吗?

1 个答案:

答案 0 :(得分:37)

是。该异常处于活动状态,直到它被捕获,并且变为非活动状态。但是它一直存在,直到处理程序的范围结束。从标准来看,强调我的:

  

§15.1/ 4:抛出的异常临时副本的内存以未指定的方式分配,除非在3.7.4.1中说明。 只要存在为该异常执行的处理程序,临时就会持续存在。

那是:

catch(...)
{ // <--

    /* ... */

} // <--

在这些箭头之间,您可以重新抛出异常。只有当处理程序范围结束时,异常才会消失。

事实上,在§15.1/ 6中,给出的示例与您的代码几乎相同:

try {
    // ...
}
catch (...) { // catch all exceptions
    // respond (partially) to exception <-- ! :D
    throw; //pass the exception to some
           // other handler
}

请注意,如果您throw没有活动例外,则会调用terminate。在处理程序中,情况可能并非如此。


如果doSomethingElse()抛出并且异常没有相应的处理程序,因为原始异常被视为已处理,新异常将替换它。 (好像它刚刚抛出,开始堆栈展开等)。

那是:

void doSomethingElse(void)
{
    try
    {
        throw "this is fine";
    }
    catch(...)
    {
        // the previous exception dies, back to
        // using the original exception
    }

    try
    {
        // rethrow the exception that was
        // active when doSomethingElse was called
        throw; 
    }
    catch (...)
    {
        throw; // and let it go again
    }

    throw "this replaces the old exception";
    // this new one takes over, begins stack unwinding
    // leaves the catch's scope, old exception is done living,
    // and now back to normal exception stuff
}

try
{
    throw "original exception";
}
catch (...)
{
  doSomethingElse();
  throw; // this won't actually be reached,
         // the new exception has begun propagating
}

当然,如果没有任何投掷,将会到达throw;并且您将按预期抛出捕获的异常。