Mina中使用多个filter,解码的时候总是出现问题,解码顺序似乎一直有问题。看了源码终于找到问题所在。
ProtocolCodecFilter 中 有一些AttributeKey
private final AttributeKey ENCODER = new AttributeKey(ProtocolCodecFilter.class, "encoder");
private final AttributeKey DECODER = new AttributeKey(ProtocolCodecFilter.class, "decoder");
private final AttributeKey DECODER_OUT = new AttributeKey(ProtocolCodecFilter.class, "decoderOut");
private final AttributeKey ENCODER_OUT = new AttributeKey(ProtocolCodecFilter.class, "encoderOut");
我原本以为 新new出来的AttributeKey,在Iosession的Map中会是不同的key,但是我错了。
public int hashCode() {
int h = 17 * 37 + ((name == null) ? 0 : name.hashCode());
return h;
}
AttributeKey 的hashCode方法被重写了,所以hash值都是根据name来的。
这样导致如果多个ProtocolCodecFilter 这些 key所在value被共享了。这样问题来了
看下DECODER_OUT这个key的value:
private static class ProtocolDecoderOutputImpl extends
AbstractProtocolDecoderOutput {
public ProtocolDecoderOutputImpl() {
// Do nothing
}
public void flush(NextFilter nextFilter, IoSession session) {
Queue<Object> messageQueue = getMessageQueue();
while (!messageQueue.isEmpty()) {
nextFilter.messageReceived(session, messageQueue.poll());
}
}
}
当第一个filter××× 解码产生一个对象A,放入messageQueue 然后调用
while (!messageQueue.isEmpty()) {
nextFilter.messageReceived(session, messageQueue.poll());
}
第二个filter×××根据传入的对象A,解码产生对象B,放入messageQueue 。
这个时候问题来了,!messageQueue.isEmpty() 又变true了循环继续,
传入第二个filter×××的是刚刚自己产生的对象B,这样显然不能再解码一次对象B。
转载于:https://blog.51cto.com/maosz1982/961688