C++是否有现有的方法来检查对象是否是派生类型对象

Does C++ have existing method to check if an object is a derived-typed object?

本文关键字:对象 类型 是否是 派生 检查 方法 是否 C++      更新时间:2023-10-16

C++是否有现有的方法来检查对象是否是派生类型对象?例如

class A
{};
class B : public A
{};
A* p;

并检查p是否指向B.

如果类是多态的(即至少有一个虚拟成员函数),则可以使用 dynamic_casttypeid

否则,没有。跟踪对象的动态类型是有代价的,并且该语言旨在避免悲观不需要它的代码。

并检查p是否指向B.

如果类中至少有一个 virtual 成员函数,则可以使用 dynamic_cast。通常使析构函数virtual

class A
{
   virtual ~A() {}
};

然后

B* bPtr = dynamic_cast<B*>(p);
if ( bPtr )
{
   // Use the pointer
}

C++是否有现有的方法来检查对象是否是派生类型对象?

实际上有两种方法可以实现此目的:

A* p = new B(); 
B* pB = static_cast<B*>(p); // Checks if B is related to A in an inheritance 
     // ^^^^^^^^^^^^^^^^^^^    hierarchy. Fails to compile if not.

A* pA = new B();
B* pB = dynamic_cast<B*>(pA); // Checks if pA actually points to an instance of B
     // ^^^^^^^^^^^^^^^^^^^^     at runtime, and returns nullptr if not
if(pB) {
  // do stuff with B
}

后一个示例要求您具有virtual基类:

class A { 
public:
    virtual ~A() {} // <<<<<<<<<<<<<<<<<<
};

如果所讨论的对象不是多态类,则可以在编译时确定指向的类对象是否具有特定的基类。这有时会发生在模板代码中,其中可能有不同的基类。

使用 std::

is_base_of 如下所示: 请注意,您还必须使用 std::remove_reference,因为 *p 是一个左值,而 decltype() 产生一个引用。

#include  <type_traits>
#include <iostream>
class A {};
class B : public A{};
int main() {
    A a;
    B b;
    A* pa=&a;
    B* pb=&b;
    std::cout << std::is_base_of<A, B>::value << "n";  // true
    std::cout << std::is_base_of<B, A>::value << "n";  // false
    std::cout << std::is_base_of<A, std::remove_reference<decltype(*pb)>::type>::value << "n";  // true
    std::cout << std::is_base_of<B, std::remove_reference<decltype(*pa)>::type>::value << "n";  // false
}