一、简介:

Hessian是一个基于Binary-RPC 实现的远程通讯library,基于 Http 协议进行传输。通过其自定义的串行化机制将请求信息进行序列化,产生二进制流。响应端根据 Hessian 提供的 API 来接收请求,根据其私有的串行化机制来将请求信息进行反序列化,传递给使用者时已是相应的请求信息对象了。处理完毕后直接返回,hessian 将结果对象进行序列化,传输至调用端。

一次完整的调用如下图所示:

二、Demo:

1. 服务端:

l  首先建立web工程,引入Hessian的jar包;

l  接着定义接口:

public interface HelloService {

public String sayHello(String name);

}

l  然后实现接口:

public class HelloServiceImpl extends HessianServlet implements HelloService {

@Override

public String sayHello(String name) {

return "Hello :"+name;

}

}

2.客户端:

public static void main(String[] args) throws MalformedURLException {

String url="http://localhost:8080/myserver/hello";

HessianProxyFactory hessianProxyFactory = new HessianProxyFactory();

// 创建接口代理程序

HelloService proxy = (HelloService) hessianProxyFactory.create(HelloService.class, url);

String result = proxy.sayHello("test");

System.out.println(result);

}

三、客户端流程分析:

1、客户端首先通过HessianProxyFactory创建HessianProxy对象,也就是使用动态代理调用服务的方法的代理对象;

2、在HessianProxy对象中缓存Method和MethodName的hash表,invoke方法中首先根据Method在hash表查找对应的MethodName,如果找到则直接使用缓存,如果未找到则根据Method读出MethodName,并缓存到hash表中;

3、HessianProxy.invoke方法中主要调用sendRequest调用远程对象,首先通过addRequestHeaders添加Http协议头信息,获取远程服务器的HessianConnection对象,然后通过AbstractHessianOutput.Call将调用方法和参数序列化为二进制,最后通过HessianConnection. sendRequest将调用信息发送到远端服务器;

4、AbstractHessianOutput.Call序列化调用信息时,startCall在二进制流里面写入起始标记“c”、版本信息以及长度等信息,然后调用Serializer对象将调用方法和参数进行序列化,最后通过写入结束标记“z”完成二进制流的序列化;

四、服务器端流程分析:

1、HessianSkeleton是hessian server端的核心类,从输入流中返序列化出客户端调用的方法和参数,对服务端服务进行调用,然后把处理结果返回给客户端;

2、HessianSkeleton.invoke首先调用AbstractHessianInput.readObject读出调用方法名和参数信息,并根据这个信息从service中查找到对应的Method对象;

String methodName = in.readMethod();

int argLength = in.readMethodArgLength();

Method method;

method = getMethod(methodName + "__" + argLength);

if (method == null)

method = getMethod(methodName);

其中getMethod是从一个_methodMap的hash表中查找Method对象,而_methodMap表在服务类对象的构造函数中就已经完成了Method对象和方法名的映射:

protected AbstractSkeleton(Class apiClass)

{

_apiClass = apiClass;

Method []methodList = apiClass.getMethods();

for (int i = 0; i < methodList.length; i++) {

Method method = methodList[i];

if (_methodMap.get(method.getName()) == null)

_methodMap.put(method.getName(), methodList[i]);

}

}

3、接下来读取参数值是调用的AbstractHessianInput.readObject读取的,而这个读取又是调用的Deserializer对象的readObject读取的,其读取方法(反序列化)与客户端的序列化格式是一一对应的;

4、将调用方法名和参数值都反序列化出来后,接下来就通过动态代理调用Service服务的对应方法,将其返回值使用writeReply进行序列化后返回给调用客户端:

public void writeReply(Object o) throws
IOException

{

startReply();

writeObject(o);

completeReply();

}

其中的startReply和completeReply分别在二进制流里面写入起始标记和结束标记,writeObject将返回值序列化到二进制流里面;

Hessian总结的更多相关文章

  1. spring remoting源码分析--Hessian分析

    1. Caucho 1.1 概况 spring-remoting代码的情况如下: 本节近分析caucho模块. 1.2 分类 其中以hession为例,Hessian远程服务调用过程: Hessian ...

  2. spring与hessian整合例

    spring与hessian的简单应用实现例: 开发环境:window7 64,jdk8,tomcat8,spring4.2.5,hessian4.0 开发语言:java hessianServer端 ...

  3. 十五分钟学会用Hessian

    了解Hessian Hessian是远程调用的一种技术,和WebService类似,但不同的是较WebService而言,它更轻量级,更简单,更快速.关于Hessian更详细全面的介绍可以查看http ...

  4. Spring远程调用技术<2>-Hessian和Burlap

    上篇谈到RMI技术,加上Spring的封装,用起来很方便,但也有一些限制 这里的Hessian和Burlap解决了上篇提到的限制,因为他们是基于http的轻量级远程服务. Hessian,和RMI一样 ...

  5. Xml,Json,Hessian,Protocol Buffers序列化对比

    简介 这篇博客主要对Xml,Json,Hessian,Protocol Buffers的序列化和反序列化性能进行对比,Xml和Json的基本概念就不说了. Hessian:Hessian是一个轻量级的 ...

  6. java和c#使用hessian通信

    介绍 hessian主页:http://hessian.caucho.com/ 一个简单的例子学习hessian服务:服务端为Java,客户端为C#. 先要准备好C#和Java的第三方类库:http: ...

  7. Hessian怎样实现远程调用

    1.Spring中除了提供HTTP调用器方式的远程调用,还对第三方的远程调用实现提供了支持,其中提供了对Hessian的支持. Hessian是由Caocho公司发布的一个轻量级的二进制协议远程调用实 ...

  8. com.caucho.hessian.io.HessianProtocolException: is unknown code 解决方案

    问题: Cannot access Hessian remote service at [http://....../remote/syllabusService]; nested exception ...

  9. Hessian最佳实践

    前言:本文主要介绍‘独立的Hessian技术’与‘结合Spring技术’的两种Hessian接口开发模式及代码示例. 一.独立的Hessian技术开发步骤 Hessian-Java服务器端必须具备以下 ...

  10. RPC hessian简单案例

    RPC(Remote procedure call) 远程服务调用. dubbox就是RPC框架,hessian是简单的RPC实现. 首先需要有接口及其实现类: 接口. public interfac ...

随机推荐

  1. 数据库别名AS区别

    Oracle之别名小结 MySQL表别名.字段别名注意事项 字段别名:可加 as  ,也可以不加,可以加单|双引号,也可以不加: 表别名:可加 as ,也可以不加,但是一定不能加单|双引号! Orac ...

  2. .NET 日期数据的格式化方法

    .HtmlEncode="False" .DataFormatString="{0:d}" C#格式化日期时间 DateTime dt = DateTime.N ...

  3. matlab xml文件交互

    xml文件以文档对象模型表示,简称DOM(Document Object Model).在Matlab中,使用xmlread读取xml文件成DOM节点,对xml文件的操作转化成对DOM节点的操作,使用 ...

  4. C#清理所有正在使用的资源

    namespace QQFrm{    partial class Form1    {        /// <summary>        /// 必需的设计器变量.        ...

  5. __new__() 与__init__()的区别

    __new__作用于__init__之前.前者可以决定是否调用后者,或者说可以决定调用那个类的__init__方法. 首先要知道在面向对象编程中,实例化基本遵循创建实例对象,初始化实例对象,最后返回实 ...

  6. Java虚拟机的组成

    Java虚拟机主要分为以下五个区: 一.方法区: 1. 有时候也成为永久代,在该区内很少发生垃圾回收,但是并不代表不发生GC,在这里进行的GC主要是对方法区里的常量池和对类型的卸载 2. 方法区主要用 ...

  7. vue框架搭建

     1到网上下载node.js,安装,(新版node,包括了npm ).2下载Git安装.3.你需要的地方建一个文件夹.打开cmd,跳转到这个文件夹输入npm install -g vue-cli 完成 ...

  8. vue mapbox 地图 demo

    执行以下命令: npm install --save mapbox-gl// cnpm install --save mapbox-gl <template> <div style= ...

  9. 如何在VMware中安装Linux系统

    这篇文章主要讲述如何在VMware12中安装RHEL6.9Linux操作系统 步骤一: 打开VMware软件,在主页中点击创建新的虚拟机或者点击左上角文件,在列表中点击新建虚拟机,如图: 步骤二: 点 ...

  10. ubuntu下载超快的一个站点

    http://mirrors.163.com/ubuntu-releases/14.04/