作为我自己的练习,我正在将一个示例程序翻译成各种语言。从C#开始,我有一个访问者模式界面,如下所示:
interface Visitor
{
void Accept(Bedroom x);
void Accept(Bathroom x);
void Accept(Kitchen x);
void Accept(LivingRoom x);
}
转移到ruby(或python,或其他),我不再根据类型从编译器调度。我应该在访客本身进行调度吗?
class Cleaner
def accept(x)
acceptBedroom(x) if Bedroom === x
acceptBathroom(x) if Bathroom === x
acceptKitchen(x) if Kitchen===x
acceptLivingRoom(x) if LivingRoom===x
end
...
或者我应该在房间的不同专业中进行调度:
class Bathroom<Room
def initialize(name)
super(name)
end
def accept(visitor)
visitor.acceptBathroom(self)
end
end
或者是否有另一种在动态语言中使用的首选习语?
答案 0 :(得分:3)
我的建议是使用前一种方法。每种方法都有优点和缺点。随着房间类型数量的增加,前者难以维持;随着清洁剂类型数量的增加,后者更难。
在Ruby中,你可以尝试
def accept(x)
send "accept#{x.class}".to_sym, x
end
PS:并非所有动态类型的语言都无法根据类型进行调度;有些人可以推断类型,或者失败,可以使用强制转换在重载选项中选择正确的方法。
答案 1 :(得分:3)
我会选择第二个版本。第一个看起来像访问者应该解决的那种代码气味:if if-else-if或switch-case语句。