如何摆脱这个Builder实现中的instanceof

时间:2012-08-03 09:26:23

标签: java oop design-patterns instanceof

想法

我需要创建命令。可以使用参数配置命令。 并非每个命令都可以接收相同的参数。所以有些人必须被忽略。

我有一个抽象类Command,我在其中定义了一个Builder。默认情况下,每个append参数都会抛出'UnsupportedOperationException'

public abstract class Command {

   public static abstract class CommandBuilder {

        // TODO instanceof. How to do this better?
        public CommandBuilder append(Parameter p)
            throws UnsupportedOperationException {

            if (p instanceof URLParameter)
                return append((URLParameter) p);

            if (p instanceof ActionParameter)
                return append((ActionParameter) p);

            if (p instanceof RepeatParameter)
                return append((RepeatParameter) p);

            if (p instanceof TimeOutParameter)
                return append((TimeOutParameter) p);

            return this;

        }

        public CommandBuilder append(URLParameter p)
                throws UnsupportedOperationException {

                    throw new UnsupportedOperationException(
                        "URLParameter not applicable");

        }

        public CommandBuilder append(RepeatParameter p)
            throws UnsupportedOperationException {

                throw new UnsupportedOperationException(
                    "RepeatParameter not applicable");

            }
            ...

}

如果您希望某个参数适用于某个具体的Command,请说一个FTPCommand。

你必须做这样的事情:

public class FTPCommand extends Command {

    public static class Builder extends CommandBuilder {

    @Override
    public CommandBuilder append(URLParameter p) {
            System.out.println("URLParemeter appended");
                return this;
            }
        }

}

因此,当提供URLParameter时,它不再抛出异常,而是应用它。

但是这个CommandBuilder的客户端可能无法提供具体的子类。所以通常给出“参数”。但它需要去正确的地方(方法)

类似于URLParameter必须到达append(UrlParameter p)

我怎样才能以干净(呃)和漂亮(r)的方式做到这一点?因为我真的不喜欢使用instanceof。

2 个答案:

答案 0 :(得分:7)

这看起来像是经典的double-dispatchvisitor情景。从双重调度参考:

  

一种将函数调用分派给不同具体的机制   函数取决于所涉及的两个对象的运行时类型   电话

ParameterCommandBuilder需要在自己之间进行互动。

CommandBuilder可以回调参数。 Parameter个对象都实现了一个公共接口,每个子类的实现会有所不同。

public CommandBuilder append(Parameter p) {
   // the append method called depends on the underlying type of 'p'
   p.append(this);
}

答案 1 :(得分:4)

我会在您的界面中添加访问者方法

interface Parameter {
    public void append(CommandBuilder builder);
}

class CommandBuilder {
    public CommandBuilder append(Parameter p) {
       p.append(this);
    }
}
相关问题