C++-恢复模板或强制转换为模板

C++ - Recover Template or cast to Template

本文关键字:转换 恢复 C++-      更新时间:2023-10-16

考虑以下代码列表:

#include <iostream>
#include <typeinfo>
class Interface { };
template<typename T>
class Class : public Interface { };
template<typename T>
Interface *producer() {
    std::cout << "produced " << typeid(T).name();
    return new Class<T>();
}
template<typename T>
void consumer(Class<T> *class_value) {
    std::cout << "consumed " << typeid(T).name();
}
void mediator(Interface *value) {
    /* The magic happens here */
    consumer(class_value);
}
int main() {
    Interface *value = producer<int>();
    mediator(value);
}

是否有任何可能的方法从"中介"函数调用"消费者"模板函数?

如果你稍微改变一下你的设计,你可以让派生类完成任务:

class Interface
{
    virtual void consume() = 0;
    virtual ~Interface() {}
};
void mediator(Interface * value)
{
    value->consume();
}
template <typename T>
void consumer(Class<T> * class_value)
{
    std::cout << "consumed " << typeid(T).name();
}
template <typename T> class Class : public Interface
{
    virtual void consume() override
    { 
        consumer<Class>(this);
    }
    // ...
};

直接答案

看起来你发布的代码是为了非法使用演员阵容而写的,添加一个合适的案例似乎是直接的答案。

void mediator(Interface *value) {
    /* The magic happens here */
    Class<int> *class_value(static_cast<Class<int> *>(value));
    consumer(class_value);
}

在这种情况下,由于已知只有int版本会到达,我们可以使用static_cast

无法缩放

然而,如果Class专门用于更多类型,则这种方法不能很好地扩展。

例如,要使mediator也处理float,您需要尝试以下每种类型:

void mediator(Interface *value) {
    /* The magic happens here */
    {
        Class<int> *class_value(dynamic_cast<Class<int> *>(value));
        if(class_value) {
            consumer(class_value);
            return;
        }
    }
    {
        Class<float> *class_value(dynamic_cast<Class<float> *>(value));
        if(class_value) {
            consumer(class_value);
            return;
        }
    }
}

双重调度

按照Kerrek SB的建议改变设计,使用双重调度,可以提供更清洁、可扩展的解决方案。

您可以使用dynamic_cast:

consumer(dynamic_cast<producer<int>&>(value));