1

我正在寻找有关实现基本消息工厂的一些想法,该工厂从输入流中读取标头并根据消息标头中定义的类型创建适当的消息类型。

所以我有类似的东西(大致......如果这里提供更好的范例,我愿意改变设计)

class MessageHeader { 
   public String type;
}

class MessageA extends Message {
   public static final String MESSAGE_TYPE = "MSGA";
   public MessageA (DataInputStream din) {
      var1 = din.readInt ();
      var2 = din.readInt ()
      // etc
   }
} 

我基本上想做这样的事情:

MessageHeader header = ... read in from stream.

if (header.type == MessageA.MESSAGE_TYPE) {
   return new MessageA (din);
} else if (header.type == MessageB.MESSAGE_TYPE) {
   return new MessageB (din);
}

虽然这个方案有效,但我觉得可能有更好的方法使用 Map 和 Interface 以某种方式......

public interface MessageCreator {
  public Message create (DataInputStream);
}

Map <String, MessageCreater> factory = new Map <String, MessageCreator> ();
factory.put (MessageTypeA.MESSAGE_TYPE, new MessageCreator () { 
                          public Message create (DataInputStream din) {
                              return new MessageA (din); }});
...
// Read message header
Message createdMessage = Map.get (header.type).create (din);

但是,每当我想使用该消息时,我都必须使用 instanceof 并转换为正确的子类。

有第三个(更好的?)选项吗?也许有一种方法可以使用模板来实现这一点。任何帮助表示赞赏。谢谢

编辑:我想重要的是要注意我想将消息“分派”到一个函数。所以本质上我真的很想这样做:

MessageHeader header = ... read in from stream.

if (header.type == MessageA.MESSAGE_TYPE) {
   handleMessageA (new MessageA (din));
} else if (header.type == MessageB.MESSAGE_TYPE) {
   handleMessageB (new MessageB (din))
}

因此,结合工厂和调度的模式将是完美的

4

1 回答 1

1

让创建消息的人真正分派给处理程序怎么样。

因此,您将添加一个像这样的处理程序接口:

public interface MessageHandler {
    void handleTypeA(MessageA message);
    void handleTypeB(MessageB message);
}

然后,您将拥有一个与您的 MessageCreator 基本相同的调度程序,除了它在处理程序上调用正确的方法而不是返回消息对象。

public interface MessageDispatcher {
    void createAndDispatch(DataInputStream input, MessageHandler handler);
}

然后,实现几乎与您发布的第一个代码片段相同:

public void createAndDispatch(DataInputStream input, MessageHandler handler) {
    MessageHeader header = ... read in from stream.

    if (header.type == MessageA.MESSAGE_TYPE) {
       handler.handleTypeA(new MessageA (din));
    } else if (header.type == MessageB.MESSAGE_TYPE) {
       handler.handleTypeB(new MessageB (din));
    }
}

现在你在代码中只有一个地方你必须做一个 switch 或 if/else if 和之后所有的东西都是专门输入的,没有更多的转换。

于 2010-10-12T04:23:06.970 回答