眾所周知,使用visitor模式的前提是目標類定義了訪問器基類及其接受訪問器的訪問接口Accept(xxxV);如果目標類未定義上述訪問器基類及訪問接口則無法使用visitor模式對其進行擴展。
如果目標類層次定義了getType函數(shù),那么可以使用另外一種手段對其擴展,具體如下:
+--------------+
| 基本擴展方案 |
+--------------+
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增加虛函數(shù)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的構(gòu)造點之后想map里面插入需要擴展的成員,并在Base的析構(gòu)點之前刪除