众所周知,使用visitor模式的前提是目标类定义了访问器基类及其接受访问器的访问接口Accept(xxxV);如果目标类未定义上述访问器基类及访问接口则无法使用visitor模式对其进行扩展。
如果目标类层次定义了getType函数,那么可以使用另外一种手段对其扩展,具体如下:
+--------------+
| 基本扩展方案 |
+--------------+
static void funcForDerivedA(Base& base, arg...)
{
DerivedA& derivedA = static_cast<DerivedA&>(base);
// use derivedA & arg... to do sth.
}
static void funcForDerivedB(Base& base, arg...)
{
DerivedB& derivedB = static_cast<DerivedB&>(base);
// use derivedB & arg... to do sth.
}
void func(Base& base, arg...) // 相当于为Base增加虚函数virtual void func(arg...)
{
static void (*funcs[])(Base& base, arg...) =
{
&funcForDerivedA,
&funcForDerivedB,
};
COMPILE_TIME_ASSERT(sizeof(funcs)/sizeof(*funcs)==CONCREATE_CLASS_COUNT);
(*funcs[base.getType()])(base, arg...);
}
+--------------+
| 扩展工厂方法 |
+--------------+
static Base* createDerivedA(arg...)
{
return new DerivedA(arg...);
}
static Base* createDerivedB(arg...)
{
return new DerivedB(arg...);
}
Base* create(Type type, arg...)
{
static void (*creators[])(arg...) =
{
&createDerivedA,
&createDerivedB,
};
COMPILE_TIME_ASSERT(sizeof(creators)/sizeof(*creators)==CONCREATE_CLASS_COUNT);
return (*creators[type])(arg...);
D}
+----------------+
| 对成员进行扩展 |
+----------------+
上述无论是基于visitor还是getType的方法都只能扩展其功能,但无法对成员进行扩展;可以采用全局映射表的方式,如下:
std::map<Base*, ExtMember>;
在Base的构造点之后想map里面插入需要扩展的成员,并在Base的析构点之前删除