为什么当我抛出派生类的对象时,基类的 catch 块会捕获异常
Why catch block of base class is catching the exception when I am throwing object of derived class?
如果我传递派生类的对象,那么应该调用派生类捕获块。但输出显示异常由基类捕获。为什么?
#include<iostream>
using namespace std;
class Base {};
class Derived: public Base {};
int main()
{
Derived d;
// some other stuff
try {
// Some monitored code
throw d;
}
catch(Base b) {
cout<<"Caught Base Exception";
}
catch(Derived d) { //This catch block is NEVER executed
cout<<"Caught Derived Exception";
}
getchar();
return 0;
}
标准说(参见工作草案的[except.handle]/4]作为示例,强调我的):
try 块的处理程序按出现顺序进行尝试。[ 注意:这使得编写永远无法执行的处理程序成为可能,例如,通过将最终派生类的处理程序放在相应的明确公共基类的处理程序之后。
这正是您在代码中所做的。
因此,我会说这是预期的行为,无论你的期望是什么。
请注意,您可以交换两个处理程序来解决问题。
关于这一点,有一点要说的。
首先,通过引用而不是值捕获异常是一种很好的做法。这可以防止在派生异常对象的情况下对其进行切片。这不是特定于例外,而是继承的一般特征。
这里讨论了这种现象的描述
此外,正如另一个答案所指出的,如果你坚持以这种方式做事,你需要改变捕获的顺序,因为派生类满足基本捕获块。还有另一种方法可以显式使用此属性:只需让基类捕获所有内容。您可以使用它,除非派生异常有一个非常具体的功能需要单独处理<</p>
catch 子句按顺序检查 - 你得到第一个匹配,而不是最佳匹配。
因此,首先检查派生类型会捕获它。
相关文章:
- std::具有相同基类的类的变体
- 是否可以初始化不可复制类型的成员变量(或基类)
- 在C++中,是否可以基于给定的标识符创建基类的新实例,反之亦然
- 基类中的函数名称解析
- C++初始化基类
- 如何通过派生类函数更改基类中的向量
- 如何定义一个纯抽象基类
- 如何使用基类指针引用派生类成员
- 继承:构造函数,初始化C++11中基类的类C数组成员
- 使用基类指针创建对象时,缺少派生类析构函数
- 如何引用基类的派生类?
- 如果基类包含双指针成员,则派生类的构造函数
- 在模板基类中为继承类中的可选重写生成虚拟方法
- 为什么此派生对象无法访问基类的后递减方法?
- 公开最直接的基类模板名称
- 当基类是依赖类型时,这是一个缺陷吗
- 如何使基类的运算符对基类的可变参数数可见(请参阅下面的代码)?
- 基类catch不捕获异常,即使它出现在派生类catch之前
- 为什么当我抛出派生类的对象时,基类的 catch 块会捕获异常
- 为什么基类catch块捕获派生类对象