通过前面的实例,我们已经了解了Hessian的大多数功能,知道Hessian可以将Java实例序列化为字节流,同时可以将字节流反序列化为Java实例,并且快速高效,下面我们就来看一看Java实例在字节流中是如何编码的,即Hessian协议格式。
下面的协议格式都是基于Hessian 1的,Hessian 2的格式大致相差不是太多,可以参考Hessian 1到官网进行学习。
Hessian 1参考Hessian 1 Protocol,Hessian 2参考Hessian 2 Protocol。
- 调用方法
调用一个方法,主要是封装Hessian版本信息,调用的方法名称以及对应的参数信息,格式如下:
'c' 1(byte) 0(byte) 'm' length(short) method_name [params] 'z'
注释: 'c'字符表示一个调用方法编码的开始;1和0表示Hessian的版本号,各占1 byte;length表示方法名称的长度,占2 byte;方法名称之后是参数列表的序列化内容,具体参数个数与调用的函数匹配,也可以没有参数,因此参数是可选的;'z'字符表示一个调用方法编码的结束标识符;
具体实例:
ByteArrayOutputStream out = new ByteArrayOutputStream();
HessianOutput hOut = new HessianOutput(out);
hOut.call("getName", new Object[]{"0000000009", 1});
hOut.flush();
byte[] bytes = out.toByteArray();
hOut.close();
FileUtils.writeByteArrayToFile(new File("/tmp/hessian/out.bin"), bytes); // 将字节输出到文件中,可以使用二进制工具查看输出内容;
利用BinaryViewer打开上面的二进制文件,可以看到下面的内容:
63 01 00 6D 00 07 67 65 74 4E 61 6D 65 ...... 7A
解释:63为'c'字符的编码;01 00为Hessian协议版本;6D为'm'字符编码;00 07为函数名称的长度,因为'getName’函数的长度为7;67 65 74 4E 61 6D 65即为'getName'函数名称的字符编码;7A为'z'的字符编码;
- primitiv