C ++是否有现有的方法来检查对象是否是派生类型的对象?例如,
class A
{};
class B : public A
{};
A* p;
并检查p
是否指向B
。
答案 0 :(得分:10)
如果类是多态的(即至少有一个虚拟成员函数),则可以使用dynamic_cast
或typeid
。
否则,不。跟踪对象的动态类型会产生成本,并且该语言旨在避免对不需要它的代码进行悲观化。
答案 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
}