我正在从事一个项目,我正在将十年前编写的旧 java 1.2 代码转换为 java 7。该项目大量(过度)使用特定的访问者。为了在概念上保持简单,假设访问者是这样的:
public interface RobotVisitor {
public Object visitHead(Head p, Object arg);
public Object visitNeck(Neck p, Object arg);
public Object visitShoulder(Shoulder p, Object arg);
public Object visitArm(Arm p, Object arg);
public Object visitHand(Hand p, Object arg);
public Object visitFinger(Finger p, Object arg);
public Object visitChest(Chest p, Object arg);
public Object visitLeg(Leg p, Object arg);
public Object visitFoot(Foot p, Object arg);
public Object visitToe(Toe p, Object arg);
// A lot of methods.
}
类Head
, Neck
, Shoulder
,Arm
等都是BodyPart
抽象类的子类,如下所示:
public abstract class BodyPart {
// A lot of fields, so it is not simple to convert this to an interface.
public abstract Object accept(RobotVisitor visitor, Object arg);
}
// All the subclasses of BodyPart are implemented like this:
public class Arm extends BodyPart {
// Some fields, getters and setters...
public Object accept(RobotVisitor visitor, Object arg) {
return visitor.visitArm(this, arg);
}
}
这些BodyPart
s 是分层的。一些BodyPart
s 可能包含其他一些BodyPart
s。但它们有时可能包含其他内容。
该访问者有几个非常不同的实现,并且正如预期的那样,代码因强制转换而瘫痪。我尝试使用泛型:
public interface RobotVisitor<R, E> {
public R visitHead(Head p, E arg);
public R visitNeck(Neck p, E arg);
public R visitShoulder(Shoulder p, E arg);
public R visitArm(Arm p, E arg);
public R visitHand(Hand p, E arg);
public R visitFinger(Finger p, E arg);
public R visitChest(Chest p, E arg);
public R visitLeg(Leg p, E arg);
public R visitFoot(Foot p, E arg);
public R visitToe(Toe p, E arg);
// A lot of methods.
}
但这不起作用。应用程序传递来自不同类型的参数,并期望同一访问者中的每组方法有不同的返回。所以,我以这样的方式结束:
public interface RobotVisitor<A, B, C, D, E, F, G, H, I> {
public A visitHead(Head p, B arg);
public A visitNeck(Neck p, B arg);
public A visitShoulder(Shoulder p, C arg);
public A visitArm(Arm p, C arg);
public A visitHand(Hand p, C arg);
public D visitFinger(Finger p, E arg);
public F visitChest(Chest p, B arg);
public A visitLeg(Leg p, G arg);
public A visitFoot(Foot p, G arg);
public H visitToe(Toe p, I arg);
// A lot of methods.
}
这只会使泛型成为一种荒谬的矫枉过正,使接口非常难以使用。
我试图将接口划分为子接口,对期望相同参数和相同返回类型并且在某些地方有效的方法进行分组,但缺点是将accept
方法从BodyPart
类中删除到对类似BodyPart
s 进行分组的子类。
然后,我遇到了一个很大的失败,有一个特定的访问者实现有一个带有BodyPart
-typed 参数的方法调用其中的accept
方法。由于我不再拥有accept
超类中的 ,这显然是一种糟糕的方式。
访问者的不同实现差异很大,访问者中的参数和返回类型也是如此。有时参数和返回类型是BodyPart
s,有时是Void
,有时是String
和Integer
,有时是摆动组件,有时是其他不相关的对象。然而在每个访问者中,访问相似BodyPart
s 的方法往往会获得相似的参数和返回类型。
客户端代码总是只调用accept
in ,仅此Head
而已。所有其他accept
方法都是从访问者调用到它自己的。
我应该怎么做才能尝试使该接口通用而不将其转换为泛型过度杀伤?instanceof
现在,我只是在寻找普通 ' 的方法中添加了很多' BodyPart
,这完全违背了使用访问者模式的全部意义。