7

我想要一个接口 ModelGenerator,它有一个方法 generate(),它接受一个可迭代的证据列表并创建一个模型。使用 STL 伪鸭类型迭代器习惯用法...

template<class Model>
class ModelGenerator {
  public:
    template<class Iterator>
    virtual bool generate(Iterator begin, Iterator end, Model& model) = 0;
};

但是虚函数不能被模板化。所以我必须对整个班级进行模板化:

template<class Model, class Iterator>
class ModelGenerator {
  public:
    virtual bool generate(Iterator begin, Iterator end, Model& model) = 0;
};

理想情况下,我想做的是这样的......

template<class Model, class Evidence>
class ModelGenerator {
  public:
    virtual bool generate(iterator<Evidence>& begin,
                          iterator<Evidence>& end,
                          Model& model) = 0;
};

但是没有迭代器继承的接口。(std::iterator 类只包含一堆 typedef,没有方法。)

我能想到的唯一方法是给 ModelGenerator 一个方法 addEvidence(),在调用 generate() 之前将它们一个一个添加,但是我必须给 ModelGenerator 状态,这有点痛苦。

如何编写一个采用任何 STL 容器的虚拟方法?

4

2 回答 2

5

你似乎需要一个any_iterator. 这是一个执行类型擦除以将您与实际迭代器实现隔离的迭代器。

Adobe 有一个实现any_iteratorhttp ://stlab.adobe.com/classadobe_1_1any__iterator.html

Boost有一个实现any_range: http: //www.boost.org/doc/libs/1_49_0/libs/range/doc/html/range/reference/ranges/any_range.html

于 2012-05-25T23:10:02.657 回答
0

为此,您可以考虑使用模板专业化而不是虚拟方法。据我了解,您有一个独特的证据类,许多不同的模型类,并且想要创建一个通用工厂以从一系列证据中生成选定的模型。

#include <vector>
#include <iostream>

struct Model1 { };
struct Model2 { };
struct Evidence { };

template<class Model>
struct ModelGenerator;

template<>
struct ModelGenerator<Model1>
{
    typedef Model1 model_type;

    template<class Iterator>
    model_type generate(Iterator begin, Iterator end)
    {
        std::cout << "Generate Model1\n";
        return model_type();
    }
};

template<>
struct ModelGenerator<Model2>
{
    typedef Model2 model_type;

    template<class Iterator>
    model_type generate(Iterator begin, Iterator end)
    {
        std::cout << "Generate Model2\n";
        return model_type();
    }
};

template<class Model, class Iterator>
Model make_model(Iterator begin, Iterator end)
{
    ModelGenerator<Model> gen;
    return gen.generate(begin, end);
}

你可以这样使用它:

int main()
{
    std::vector<Evidence> v;

    Model1 m1 = make_model<Model1>(v.begin(), v.end());
    Model2 m2 = make_model<Model2>(v.begin(), v.end());
}
于 2012-05-26T03:51:36.830 回答