前面我们讲了对如下的数据协议使用“模板方法模式”来解析包头和包尾的过程:
现在接着来说一说对包体的解析。
这里拿出三个数据包来作为例子:
-
初始化命令
这个命令的包体如下:这个命令包体很简单,只有一个字节,表明是“初始化命令”。
-
查版本号命令
这个命令的包体如下:这个命令包体相对复杂一点,一共3个字节。
-
读钱箱电子ID命令
这个命令的包体如下:这个命令有4个字节。
从上述包体格式可以看出,包体开头都有一个字节来标识命令类型,然后才是命令体内容。
对于这样的包体格式,特别适合使用“责任链模式”,下面来详细说一说如何使用该模式做encode。
首先是父类的实现:
public class CashboxBodyEncoder {
private CashboxBodyEncoder next;
public CashboxBodyEncoder(CashboxBodyEncoder next) {
this.next = next;
}
责任链上的每一个节点,都必须初始化下一个节点,如上述代码中 next
所示。
public void doEncoder(CashboxBaseData baseData, ByteBuf out)
{
if (this.next != null) this.next.doEncoder(baseData, out);
}
}
doEncoder
方法是做encode工作的方法,它来判断当前节点有没有下一个节点,如果有,则把工作给下一个节点做。
以上就是“责任链模式”的父类。
下面来看看子类实现。
首先是“初始化”命令:
@Slf4j
public class CashboxInitialiseEncoder extends CashboxBodyEncoder{
public CashboxInitialiseEncoder(CashboxBodyEncoder next) {
super(next);
}
前面说了,每一个子类在初始化的时候,都需要指定下一个节点对象。
@Override
public void doEncoder(CashboxBaseData baseData, ByteBuf out) {
覆盖父类的doEncoder
方法。
if (baseData.getType() == INITIALISE)
{
log.info("进入初始化命令写入...");
}
首先判断,如果是“初始化”命令,则做encode。这里“初始化”命令的命令类型,已经在encode包头的时候顺手做了,所以,在做“初始化”命令的encode的时候,没有任何事情可以做。
else super.doEncoder(baseData, out);
}
}
如果不是“初始化”命令,则交给父类的doEncoder
方法处理,而它的处理是将数据交给下一个节点处理,这就是责任顺着责任链往下传的意图。
下面,我们来看查看版本号命令的实现:
@Slf4j
public class CashboxVersionReadResultDecoder extends CashboxBodyDecoder{
public CashboxVersionReadResultDecoder(CashboxBodyDecoder next) {
super(next);
}
@Override
public CashboxBaseData decode(CashboxBaseData baseData, ByteBuf buffer, CashboxParamHandleType handleType) {
if (baseData.getType() == CASHBOX_VERSION)
{
log.info("解析固件版本号结果...");
CashboxVersionReadResultData data = new CashboxVersionReadResultData((CashboxBaseResultData) baseData);
data.setVersion(buffer.getCharSequence(RESULT_PORT_BEGIN, data.getLength() - SIZE_OF_DATA_4_CODE, Charset.forName("utf-8")).toString());
return data;
}
else return super.decode(baseData, buffer, handleType);
}
}
读电子钱箱ID命令的代码就不再给出了,感兴趣的话,大家可以自己尝试写写。
最后是实例化各命令类的代码:
public class CashboxBodyDecoderFactory {
public static CashboxBodyDecoder getBodyDecoder()
{
return new CashboxResult4AckDecoder(
new CashboxResult4NakDecoder(
new CashboxSelfcheckDataDecoder(
new CashboxParaDataDecoder(
new CashoutResultDataDecoder(
new CashboxParamResultDataDecoder(
new CashboxCoreStateResultDecoder(
new CashboxVersionReadResultDecoder(
new CashboxInitialiseResultDecoder(
new CashboxStateResultDataDecoder(null))))))))));
}
}
可以看到,除了最后一个节点,每一个节点都有下一个节点,到了最后一个节点,责任的传递才会中止。
其类图如下所示: