我正在检查dynamic_cast的行为,发现当它失败时,仅当目标是引用类型时才抛出std :: bad_cast异常。如果目标是指针类型,则不会从强制转换中抛出异常。这是我的示例代码:
class A
{
public:
virtual ~A()
{
}
};
class B : public A
{
};
int main()
{
A* p = new A;
//Using reference
try
{
B& b = dynamic_cast<B&>(*p);
}
catch(std::bad_cast exp)
{
std::cout<<"Caught bad cast\n";
}
//Using pointer
try
{
B* pB = dynamic_cast<B*>(p);
if( pB == NULL)
{
std::cout<<"NULL Pointer\n";
}
}
catch(std::bad_cast exp)
{
std::cout<<"Caught bad cast\n";
}
return 0;
}
输出为“Caught bad cast”和“NULL pointer”。代码使用VS2008编译。这是正确的行为吗?如果是,那为什么会有差异?
答案 0 :(得分:63)
是的,这是正确的行为。原因是你可以有一个空指针,但不能是空引用 - 任何引用都必须绑定到一个对象。
因此,当指针类型的dynamic_cast失败时,它返回一个空指针,调用者可以检查它,但是当它为引用类型失败时,它不能返回空引用,所以异常是唯一合理的方法发出问题。
答案 1 :(得分:24)
参见C ++标准,第5.2.7 / 9节:
9失败演员的价值 指针类型是空指针值 所需结果类型。失败了 强制转换为引用类型会抛出bad_cast (18.5.2)。
至于为什么 - 这些是来自D&amp; S的Stroustrup的话。电子书,第14.2.2节:
我想要一个参考演员 关于参考类型的假设 检查并认为它失败了 我的假设是错的。如果相反 我想选择合理的 替代方案,我使用指针演员和 测试结果。
答案 2 :(得分:8)
是,5.2.7 / 9
失败的强制转换为指针类型的值是所需结果类型的空指针值。失败的强制转换为引用类型会抛出bad_cast(18.5.2)。
答案 3 :(得分:5)
是的,确实如此。因为dynamic_cast
不能为失败的引用转换返回NULL,所以异常是唯一的出路。
也就是说,引用不能为NULL,因此没有什么适合返回。