c++ - 是否有可能 "copy"一个模板化类派生自非模板化基类的对象?

标签 c++ class templates polymorphism

我用 C++ 构建了一个小型、范围有限的跨平台 UI 库。它使用以下类来处理 UI 回调:

// Base class for templated __Callback class to allow for passing __Callback
// objects as parameters, storing in containers, etc. (as Callback*)
class Callback
{
    public:
        virtual void execute() = 0;
        virtual ~Callback() { }

    protected:
        Callback() { }
};
正如评论所描述的,这是回调的基类 - 它允许将它们作为参数传递并将它们存储在 UI 小部件中,以便可以执行正确的回调(例如,当用户单击按钮时)。

// C++ has no callbacks (in the sense of an action [method] of a
// target [object]), so we had to roll our own. This class can be used
// directly, but the NEW_CALLBACK macro is easier.
template <class __TargetType>
class __Callback : public Callback
{
    public:
        typedef __TargetType TargetType;
        typedef void (TargetType::*ActionType)(void);

        virtual void execute()
        {
            (this->__target->*this->__action)();
        }

        __Callback(TargetType* target_, ActionType action_) :
        Callback(), __target(target_), __action(action_) { }

        virtual ~__Callback() { }

    private:
        // target object for the callback
        TargetType* __target;

        // action (method) of the target that will be called
        ActionType __action;
};

这个模板类是回调范例的核心。它存储一个指向对象的指针和一个指向成员函数的指针,以便稍后可以在目标对象上调用成员函数。

#define NEW_CALLBACK(class_, obj_, act_) \
    new __Callback<class_>(obj_, &class_::act_)

这个宏只是让创建模板化 __Callback 对象变得更容易。

这已经运行了很长一段时间了!带有回调的按钮可能会被实例化为:

MyClass* obj = new MyClass();
Button* btn = new Button("Title", NEW_CALLBACK(MyClass, obj, btnClicked));

这将创建一个按钮,稍后放置在窗口或其他容器中,单击时它将调用 obj->btnClicked()

现在,我的问题(很抱歉冗长的设置,我认为我无法比这更精简)。出现了一个情况,我需要复制一个 Callback* 对象。当然,由于它只是指向基类的指针,因此我无法确定模板化派生类的类型。

如何复制任意 Callback 对象,使拷贝指向与原始对象相同的目标和操作? 或者,是否有一种完全不同的方法对于我应该采取的回调问题(尽管我不希望对其进行太多更改)?

谢谢大家!

最佳答案

我不知道您是否应该采取更好的方法,但这似乎是克隆方法的理想用途。

您只需在 __Callback 模板中定义一个复制构造函数,在基类中定义一个纯虚拟 Clone 方法,然后在模板中实现该虚拟方法(利用您创建的复制构造函数)

例如:

class Callback
{
    public:
    ...
    virtual Callback* Clone()=0;
};

template <class __TargetType>
class __Callback : public Callback
{
public:
    ...
    __Callback(const __Callback& other) :
        __target(other.target_), __action(other.action_) { }

    virtual Callback* Clone()
    {
        return new __Callback(this);
    }
}

关于c++ - 是否有可能 "copy"一个模板化类派生自非模板化基类的对象?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10560081/

相关文章:

c++ - CRTP中的析构函数是怎么实现的?

c++ - 这个问题叫什么?如果通过基类型传递结构,则后代类型中的字段在方法中不可用

c++ - 处理隧道时,套接字 read() 函数有时从不返回 0

java - 向 ArrayList 添加一个新对象,该对象是另一个类的属性

PHP 简单 HTML DOM 解析器 : Select only DIVs with multiple classes

c++ - 从文件中读取行时出错

python - 在 Python 中为 Enum 类创建属性

c++ - 为什么编译器选项会影响模板实现的选择?

c++ - 如何避免使用特征类违反 ODR

c++ - 如何确定 C++ 对象是否为 time_t