这对于访问者设计模式是可行的。
访问者将新功能注入到类的层次结构中。这不必是虚拟的,可以毫无问题地表示为功能模板。
访问者通常的缺点适用(循环依赖),通常的修复/解决方法(非循环动态访问者技术)也是如此。
这是一个(常规的、循环的)实现示例,快速组合在一起。
#include <iostream>
#include <typeinfo>
class Manager1;
class Manager2;
class Visitor
{
public:
virtual void visit (Manager1*) = 0;
virtual void visit (Manager2*) = 0;
};
class IManager
{
public:
template<class T> void Set(T* t);
template<class T> T *Get();
virtual void accept(Visitor* v) = 0;
};
class Manager1: public IManager
{
public:
template<class T> void Set(T*)
{ std::cout << "Manager1::Set " << typeid(T).name() << std::endl; }
template<class T> T *Get()
{ std::cout << "Manager1::Get " << typeid(T).name() << std::endl; return 0; }
virtual void accept(Visitor* v)
{ v->visit(this); }
};
class Manager2: public IManager
{
public:
template<class T> void Set(T* t)
{ std::cout << "Manager2::Set " << typeid(T).name() << std::endl; }
template<class T> T *Get()
{ std::cout << "Manager2::Get " << typeid(T).name() << std::endl; return 0; }
virtual void accept(Visitor* v)
{ v->visit(this); }
};
template <class T>
class GetVisitor : public Visitor
{
public:
T* GetFunc(IManager* m) { m->accept(this); return t; }
void visit(Manager1* m) { t = m->Get<T>(); }
void visit(Manager2* m) { t = m->Get<T>(); }
private:
T* t;
};
template <class T>
class SetVisitor : public Visitor
{
public:
void SetFunc(IManager* m, T* tt) { t = tt; m->accept(this); }
void visit(Manager1* m) { m->Set(t); }
void visit(Manager2* m) { m->Set(t); }
private:
T* t;
};
template<class T> void IManager::Set(T* t)
{ SetVisitor<T> v; v.SetFunc(this, t); }
template<class T> T *IManager::Get()
{ GetVisitor<T> v; return v.GetFunc(this); }
class Foo {};
int main ()
{
IManager* mgr1 = new Manager1;
IManager* mgr2 = new Manager2;
int a = 5;
const char* b = "abc";
double c = 1.0;
Foo d;
mgr1->Set(&a);
mgr1->Set(&b);
mgr1->Set(&c);
mgr1->Set(&d);
mgr1->Get<Foo>();
mgr2->Set(&a);
mgr2->Set(&b);
mgr2->Set(&c);
mgr2->Set(&d);
mgr2->Get<Foo>();
}
可以用一些 s 来打破循环dynamic_cast
,但每个用户Set
和Get
仍将依赖于所有 Manager
类。这就是模板在 C++ 中的工作方式。如果这是不可接受的,那么模板可能不是这项工作的正确工具。