具有可变返回类型的 C++ 访问者模式

标签 c++ templates virtual-functions visitor-pattern

我有以下设置:

class IVisitor{
public:
    virtual SomeType visit(IVisitable& visitable) = 0;
};

class IVisitable{
public:
    virtual SomeType accept(IVisitor& visitor) = 0;
};

class ConcreteVisitor : public IVisitor{
public:
    SomeType visit(IVisitable& visitable) override {
        return calculateSomeStuff();
    }
};

class ConcertVisitable : public IVisitable{
public:
    SomeType accept(IVisitor& visitor) override {
        return visitor.visit(*this);
    }
};

但我希望不同的访问者返回不同的类型。为此,IVisitable::accept() 也应该返回不同的类型。

这将导致以下(不正确的)代码:

template <typename R>
class IVisitor{
public:
    virtual R visit(IVisitable& visitable) = 0;
};

class IVisitable{
public:
    template <typename R>
    virtual R accept(IVisitor<R>& visitor) = 0;
};

class ConcreteVisitorOne : public IVisitor<int>{
public:
    int visit(IVisitable& visitable) override {
        return calculateSomeStuff();
    }
};

class ConcreteVisitorTwo : public IVisitor<bool>{
public:
    bool visit(IVisitable& visitable) override {
        return calculateSomeStuff();
    }
};

class ConcertVisitable : public IVisitable{
public:
    int accept<int>(IVisitor<int>& visitor) override {
        return visitor.visit(*this);
    }
};

不幸的是,这在 C++ 中是不可能的,因为 virtual 函数 IVistitable::accept() 不能被模板化。

有人知道解决此限制的方法吗?

最佳答案

几个选项:

  • 不返回任何东西,处理访问者内部的值。访问者可以完全自由地存储或处理任何返回值和类型。

  • 返回可变类型(std::anystd::variant)。请注意,使用它们需要您找出实际类型,这可能与您原来的问题有点相似。

  • 继承多个IVisitor<>接口(interface)。这甚至可能是最简单的方法,尽管它不一定优雅。

关于具有可变返回类型的 C++ 访问者模式,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/69004879/

相关文章:

c++ - 与普通 cout 的行为相比,为什么 c++ 中的 for 循环访问未初始化的内存位置?

c++ - 如何在继承类中再次使用虚函数

c++ - 多个类型名的部分模板特化

c++ - 在 C++ 中更改对象的动态类型

C++ 多态性,不完全向下转型

c++ - OpenCV 访问 MAT 对象中的 RGB 值

c++ - TFTP 源代码示例

c++ - 类型删除 : Retrieving value - type check at compile time

c++ - MinGW 从模板调用模板

.net - C++ 与 C++/CLI : Const qualification of virtual function parameters