C++查找带有指向超类的指针的子类的类型

C++ finding type of subclass with a pointer to superclass

本文关键字:指针 子类 类型 超类 查找 C++      更新时间:2023-10-16

在我的C++代码中,我有一个指向 Light 对象的指针向量。P_Light是光的一个子类,具有场位置。对于指向实际上是P_Light的光源的每个指针,我需要对位置字段执行一些操作。我做了一些搜索,似乎我可以实现一个虚拟方法,但我不需要 Light 中的方法,因为其他类型的 Light 没有位置。我也考虑过选角,但我不知道如何做到这一点。

std::vector<Vector> light_dirs;
for(int i=0; i<lights.size; i++){
    Light *l = lights[i];   
    //cast here?
}

编辑::在不同的帖子中看到,也许使用qobject_cast是个好主意。这样看起来更好吗?

std::vector<Vector> light_dirs;
for(int i=0; i<lights.size; i++){
    Light *l = lights[i];   
    P_Light* pl = qobject_cast<P_Light*>(l);
    if(pl != nullptr) //dostuff;
}

这应该有效。

P_Light* p_light = dynamic_cast<P_Light*>(l);

检查 RTTI 和动态投射 https://en.wikipedia.org/wiki/Run-time_type_information

正如Seo指出的那样,动态强制转换可能是在不重构类结构的情况下快速适应您描述的用例的最直接方法。

作为替代方法,您可以考虑使用 Visitor 模式的实现,该模式提供编译时安全性和验证。

class P_Light;
class NP_Light;
/* abstract class providing dispatching based on 
 * whether Light objects interact with the position field */
class Light_Visiter{
  virtual ~Light_Visiter(){}
  virtual void visit(NP_light&) = 0;
  virtual void visit(P_Light&) = 0;
};
class Light{
  virtual ~Light(){}
  virtual void visit(Light_Visitor&) = 0;
  // other methods
};
class P_Light : public Light{
  void visit(Light_Visitor& lv){lv.visit(*self);}
  // other methods
};
/* Light objects which do not interact with the position
 * field inherit from NP_Light */ 
class NP_Light : public Light{
  void visit(Light_Visitor& lv){lv.visit(*self);}
  // other methods
};

这为您提供了与动态强制转换相同的优势 - 您的类仍然独立于对其进行操作的算法 - 但调度逻辑是集中的,而不是分散在整个源代码中。

有关访客模式的更多信息,请参阅这篇不错的文章。