50

我正在检查 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 编译。这是正确的行为吗?如果是,那么为什么会有差异?

4

4 回答 4

84

是的,这是正确的行为。原因是你可以有一个空指针,但不能有一个空引用——任何引用都必须绑定到一个对象。

因此,当指针类型的 dynamic_cast 失败时,它返回一个空指针,调用者可以检查它,但是当它针对引用类型失败时,它不能返回空引用,因此异常是发出问题信号的唯一合理方式.

于 2009-08-14T09:14:38.153 回答
34

请参阅 C++ 标准,第 5.2.7/9 节:

9 转换为指针类型失败的值是所需结果类型的空指针值。转换为引用类型失败会抛出 bad_cast (18.5.2)。

至于为什么——这些是 Stroustrup 在 D & E 书中第 14.2.2 节的话:

当我想检查一个关于引用类型的假设并认为它是我的假设错误的失败时,我使用引用强制转换。相反,如果我想在合理的替代方案中进行选择,我会使用指针转换并测试结果。

于 2009-08-14T09:17:49.583 回答
8

是的,5.2.7/9

转换为指针类型失败的值是所需结果类型的空指针值。转换为引用类型失败会抛出 bad_cast (18.5.2)。

于 2009-08-14T09:17:42.407 回答
7

是的。因为dynamic_cast不能为失败的引用转换返回 NULL,所以异常是唯一的出路。

也就是说,一个引用不能为NULL,所以没有什么适合返回的。

于 2009-08-14T09:14:34.877 回答