介绍
Hessian是一个轻量级的remotingonhttp工具,使用简单的方法提供了RMI的功能。早在08年有人做过测试:一个UserData类,有一个字符串属性,一个日期属性,一个double属性,分别用java,hessian来序列化一百万次,结果让人吃惊,不止是hessian序列化的速度要比java的快上一倍,而且hessian序列化后的字节数也要比java的少一倍。
序列化
序列化宏观来看就四步
- 建立进行序列化的工厂类
- 用工厂类找到要要进行序列化类对应的序列化器
- 序列化器对将进行序列化的对象进行内省
- 内省完成后调用序列化器的writeObject(writeObject中按照一定规则把数据写入byte流)
下面用Hessian简单序列化一个对象来看
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25
| import com.caucho.hessian.io.HessianOutput; import java.io.ByteArrayOutputStream; import java.io.IOException; import java.io.Serializable;
class User implements Serializable { String name; int age;
User() { age = 18; name = "cl0und"; } }
public class HessianTest { public static void main(String[] args) throws IOException { User user = new User(); ByteArrayOutputStream os=new ByteArrayOutputStream(); HessianOutput output=new HessianOutput(os); output.writeObject(user); } }
|
建立进行序列化的工厂类
把ByteArrayOutputStream放入了HessianOutput中典型的装饰器模式。
data:image/s3,"s3://crabby-images/060e1/060e14ebf9092eab3cb34b1bda442030054a8f50" alt="image.png"
新建序列工厂类(后续会通过工厂类来寻找类的对应序列化器)。
data:image/s3,"s3://crabby-images/136b4/136b4c50401dd05505a081362f88fe50a368b158" alt="image.png"
工厂类的创建过程中,在类静态块中加入了java各种类型的反序列器以及和Hessian有关的反序列器。这里和序列化没关系,只是提一嘴。data:image/s3,"s3://crabby-images/a8fd2/a8fd2687b627c5f8f51db2c1f07680aef1ca4ffd" alt="image.png"
在类初始化完成后,正式进入SerializerFactory的构造方法。可以看到SerializerFactory允许不安全的序列化,设置默认的上下文工厂类是_contextFactory,这个_contextFactory是生产序列化器的工厂。
data:image/s3,"s3://crabby-images/5c206/5c20648b54bbd16c4f22eb30914ac17dfab4daf8" alt="image.png"
data:image/s3,"s3://crabby-images/45da1/45da188ab38081c4d1ab0001b56de8e66e081e89" alt="image.png"
用工厂类找到要要进行序列化类对应的序列化器
根据传入类型获取相应的序列化器。
data:image/s3,"s3://crabby-images/7a491/7a491037bc2d469dee62b547422e77f6c05c4169" alt="image.png"
获取序列化器的规则是如果_cachedSerializerMap里面又就直接从里面取,如果没有就用loadSerializer找,找出来的加入_cachedSerializerMap避免二次加载。
data:image/s3,"s3://crabby-images/59753/597538487d980872a895b27bf87009d6f0025a0a" alt="image.png"
loadSerializer找的逻辑是先看有没有其它工厂类,如果有就用,如果没有就用_contextFactory造。
data:image/s3,"s3://crabby-images/af0a1/af0a1aa7b6b5a6ddb22090da072f22c81e9dff9f" alt="image.png"
因为User是自建类型_contextFactory造不出对应序列化器。
data:image/s3,"s3://crabby-images/c1640/c16403d8a9915a7b40e6bca16211ab34be9acb66" alt="image.png"
下一步尝试getCustomSerializer来找序列化器,看方法的意思应该是找是否有客户自己定制的序列化器。生成序列器规则是:如果_customSerializerMap有就直接返回,如果没有就按照_类名+HessianSerializer_的规则来找。很遗憾我们并没有对User做个性化定制。
data:image/s3,"s3://crabby-images/ae33b/ae33b1d9a1ba67d7690cfff9b9bfff5aa6293c28" alt="image.png"
因为没有我们没有对User做个性化的序列化器,所以最后会调用getDefaultSerializer来返回默认序列化器。又因为User既不是内置类也不是和hessian有关的特殊类但是允许了不安全的序列化,所以最后被选中的序列化器是UnsafeSerializer。
data:image/s3,"s3://crabby-images/3e861/3e86102a106dfc73d3a8ae521dfdbe58acfddbc0" alt="image.png"
序列化器对序列化对象进行内省
在构造方法中使用introspect进行内省。
data:image/s3,"s3://crabby-images/7f5e1/7f5e16b292976aac8797affb6cb6b523464cdefb" alt="image.png"
data:image/s3,"s3://crabby-images/d877b/d877b0de9d029b572d56db94ad830b28f5a356cf" alt="image.png"
introspect最后几步是在为类里面的属性找序列化器(getFiledSerializer)。
data:image/s3,"s3://crabby-images/c6b82/c6b82944c94dd21c72b7e1887972f9a56ab0e6d5" alt="image.png"
比如这里User类中name属性是String,就创建针对String的序列化器。
data:image/s3,"s3://crabby-images/9bbc7/9bbc70f981f36a9e9f790512d605dbe3fab38409" alt="image.png"
String的序列化器包含了field类及其偏移量。
data:image/s3,"s3://crabby-images/cc872/cc872f8d5b1e3b233719fe5c16f185b7c71b85e1" alt="image.png"
创建类及及其字段序列化器后一路向外传,然后调用writeObject。
data:image/s3,"s3://crabby-images/55aad/55aad6b5e28abe2a3fdbd6c543b95948f7a7d2c4" alt="image.png"
data:image/s3,"s3://crabby-images/84086/840865bb10924801468309197946f23032af5fba" alt="image.png"
writeObjectBegin 写了Object的魔术头、长度、类名
data:image/s3,"s3://crabby-images/23989/239893deb798cabc43d8f677baf206434cf11307" alt="image.png"
data:image/s3,"s3://crabby-images/59f68/59f68b4c7dc00891669f4c6ff74097e3bbef631c" alt="image.png"
writeObject10 负责序列化字段
data:image/s3,"s3://crabby-images/20ded/20ded6afc687ef8ba81b86bcf3991a9be42cd5ac" alt="image.png"
先是写要序列化的字符串魔术值、字段名长度、字段名,其中 os.write(83)
代表 S
代表字符串。data:image/s3,"s3://crabby-images/49e0c/49e0c39e759d9ee5d99562c67e04c3fff20cca26" alt="image.png"
String类型学序列化的时候
data:image/s3,"s3://crabby-images/f6bc5/f6bc56de95bb4fa8e40e9202a0140d05f135bf06" alt="image.png"
data:image/s3,"s3://crabby-images/8579a/8579acd2c1c5b071b07180be3248790ea0522c9a" alt="image.png"
int类型序列化的时候
data:image/s3,"s3://crabby-images/30140/301402cff349099a7b639338d84a5cbcc073645e" alt="image.png"
data:image/s3,"s3://crabby-images/90736/9073611a07d808ac04e9259c2078d24e562ad5b7" alt="image.png"
最后的结果
data:image/s3,"s3://crabby-images/40d72/40d72fa9b1e180fd73a2aae1aa79be198db95cea" alt="image.png"
业务场景
环境搭建
说穿了就是RPC,用实际的代码来看看
pom.xml加入
1 2 3 4 5 6 7 8 9 10 11 12 13
| <dependencies> <dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>4.11</version> <scope>test</scope> </dependency> <dependency> <groupId>com.caucho</groupId> <artifactId>hessian</artifactId> <version>4.0.60</version> </dependency> </dependencies>
|
web.xml
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20
| <!DOCTYPE web-app PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN" "http://java.sun.com/dtd/web-app_2_3.dtd" >
<web-app> <display-name>Archetype Created Web Application</display-name> <servlet> <servlet-name>HessianServlet</servlet-name> <servlet-class>com.caucho.hessian.server.HessianServlet</servlet-class> <init-param> <param-name>service-class</param-name> <param-value>org.syclover.hessian.BasicService</param-value> </init-param> </servlet>
<servlet-mapping> <servlet-name>HessianServlet</servlet-name> <url-pattern>/api/service</url-pattern> </servlet-mapping> </web-app>
|
代码忘记从网上哪里copy的了,侵删。
client/server端共有代码
1 2 3 4 5 6 7 8 9 10 11 12 13 14
| package org.syclover.hessian;
public interface IBasicApi {
public boolean setUserName(String name);
public String sayHello();
public User getUser();
}
|
client端代码
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19
| package org.syclover.hessian;
import com.caucho.hessian.client.HessianProxyFactory; import java.net.MalformedURLException;
public class ClientTest {
public static void main(String[] args) throws MalformedURLException {
String url = "http://localhost:8080/hessianTest_war_exploded/api/service"; HessianProxyFactory factory = new HessianProxyFactory(); IBasicApi api = (IBasicApi) factory.create(IBasicApi.class, url); api.setUserName("mahc"); System.out.println(api.sayHello()); System.out.println(api.getUser().getName()); System.out.println(api.getUser().getAge()); } }
|
server端代码
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24
| package org.syclover.hessian;
import org.syclover.hessian.User;
public class BasicService implements IBasicApi {
private String name;
@Override public boolean setUserName(String name) { this.name = name; return false; }
@Override public String sayHello() { return "Hello " + name + ",Welcome to Hessian!"; }
@Override public User getUser() { return new User(name, 23); } }
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36
| package org.syclover.hessian;
import java.io.Serializable;
public class User implements Serializable {
private String name; private Integer age;
public User() { super(); }
public User(String name, Integer age) { super(); this.name = name; this.age = age; }
public String getName() { return name; }
public void setName(String name) { this.name = name; }
public Integer getAge() { return age; }
public void setAge(Integer age) { this.age = age; } }
|
这里为了测试方便把c/s放到一起了
data:image/s3,"s3://crabby-images/beff3/beff30bb1ce432590f1131f591f2506f9fa35e28" alt="image.png"
进行RPC
data:image/s3,"s3://crabby-images/079de/079deb79bf350f8a8bc7a64c23fdb432be895988" alt="image.png"
参考
Hessian使用教程
JVM 类装载原理分析-ClassLoader原理分析
hessian 序列化实现 初探