如何从向量中获取真正的类实例名称?

Mic*_*ael 2 c++ inheritance vector c++11 c++14

我有课:

class IComponent{};
Run Code Online (Sandbox Code Playgroud)

我将其用作其他特定类的基类.

class First : public IComponent{public: void First();};   
class Second : public IComponent{public: void Second();};
Run Code Online (Sandbox Code Playgroud)

我有一个类Object,它有一个存储从IComponent继承的类的向量.

vector<IComponent*> vectorOfComponents;
Run Code Online (Sandbox Code Playgroud)

在这个类中,我有一个模板方法,可以向vector添加组件.

template<typename comp>
void AddComponent() {
    comp * c = new comp();
    getComponents().push_back(c);
}
Run Code Online (Sandbox Code Playgroud)

但是我也希望能够获得特定类型的组件,所以我创建了这个方法:

template<typename comp>
comp * GetComponent()
{
    comp component;
    for (int i = 0; i < GetComponentsCount(); i++)
        if (typeid(*getComponents()[i]).name() == typeid(component).name())
            return (comp*)getComponents()[i];
    return NULL;
}
Run Code Online (Sandbox Code Playgroud)

当我以后运行它:

    if (obj.GetComponent<FirstClass>() != NULL)
        std::cout << "isn't null\n";
    else
        std::cout << "is null\n";
Run Code Online (Sandbox Code Playgroud)

我知道vector可以保存IComponent继承类的推送实例,但它总是打印我为null,因为第一个typeid()显示我的IComponent,第二个显示我特定的继承组件.

如何从向量中获取特定类型?我试过多态,但是没有用,或者我做错了什么:

template<typename comp>
void AddComponent() {
    comp c = comp();
    IComponent * i = &c;
    getComponents().push_back(i);
}
Run Code Online (Sandbox Code Playgroud)

Pot*_*ter 6

typeid仅对多态类类型动态工作.对于一个多态的类,它需要一个virtual函数.此外,析构函数必须是virtual一个类对象可能由基类指针拥有,这可能是这里的情况.

所以,你应该使用,

class IComponent{
    virtual ~ IComponent() = default;
};
Run Code Online (Sandbox Code Playgroud)

编辑:同样,type_info::name()返回一个C字符串指针,使用时无法比较==.使用动态库的程序可以name在不同的地址观察相同的字符串(尽管这是不寻常的).比较type_info对象.另外,typeid可以将类型作为操作数; 您不需要创建一个对象来使用它.所以,你可以这样做,

if (typeid(*getComponents()[i]) == typeid(comp))
Run Code Online (Sandbox Code Playgroud)

请注意,这会检查完全匹配的类型.要查找派生对象(允许comp成为基类),请使用dynamic_cast:

if (comp *p = dynamic_cast<comp *>(getComponents()[i])) {
    return p;
}
Run Code Online (Sandbox Code Playgroud)