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

时间:2016-06-09 20:26:16

标签: c++ dynamic-cast

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

class A
{};

class B : public A
{};

A* p;

并检查p是否指向B

4 个答案:

答案 0 :(得分:10)

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

否则,不。跟踪对象的动态类型会产生成本,并且该语言旨在避免对不需要它的代码进行悲观化。

答案 1 :(得分:5)

  

并检查p是否指向B

如果班级中至少有一个dynamic_cast成员函数,则可以使用virtual。制作析构函数virtual

是很常见的
class A
{
   virtual ~A() {}
};

然后,

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

答案 2 :(得分:1)

  

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() {} // <<<<<<<<<<<<<<<<<<
};

答案 3 :(得分:0)

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

你使用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
}