hessian原理解析一(客户端分析)

  hessian 是一款开源的二进制远程通讯协议,使用简单方法提供了RMI功能,主要用于面向对象的消息通信。 优点:跨平台、多语言支持、使用简单 缺点:传递复杂对象性能会下降,不适合安全性高的应用

  

一 、hessian demo 示例:

  1、新建一个maven项目,包含3个模块 API 模块(远程接口,供客户端和服务端使用),客户端模块和服务端模块

  2、添加hessian依赖包

       在pom.xml 中添加hessian依赖

    <dependency>

      <groupId>com.caucho</groupId>

      <artifactId>hessian</artifactId>

      <version>4.0.38</version>

    </dependency>

  3、远程接口定义

    在hessian-api 模块定义接口

public interface HelloHessian {

      String sayHello(User user);

}

  4、服务端接口实现

在hessian-server 模块实现API 接口

  public class HelloHessianImpl implements HelloHessian {

     public String sayHello(User user) {

      String userName = user.getName();

      System.out.println("hello:"+userName);

      return userName;

    }

  }

  5、服务端配置远程服务地址

  hessian 是依赖web 容器 需在web.xml 中配置服务请求地址

  <servlet>

    <servlet-name>HelloHessian</servlet-name>

     <!-- RPC HessianServlet处理类 -->

     <servlet-class>com.caucho.hessian.server.HessianServlet</servlet-class>

    <init-param>

      <param-name>service-class</param-name>

       <!-- 远程服务实现类 -->

      <param-value>com.hessian.server.HelloHessianImpl</param-value>

     </init-param>

  </servlet>

  <servlet-mapping>

     <servlet-name>HelloHessian</servlet-name>

     <url-pattern>/HelloHessian</url-pattern>

   </servlet-mapping>

  6、客户端远程接口调用

    // 远程服务地址

String url = "http://localhost:8080/HelloHessian";

//通过hessian代理工程创建接口代理

HessianProxyFactory fatory = new HessianProxyFactory();

HelloHessian helloHessian = (HelloHessian)fatory .create(HelloHessian.class,url);

// 执行方法调用

helloHessian.sayHello(new User(10, "tiger"));

  7、运行demo

7.1、启动服务器

7.2、运行客户端程序

控制台打印 hello:tiger

二、hessian 工作流程

三、客户端源码分析

1、客户端请求时序图

2、客户端创建接口的代理类分析:

HessianProxyFactory factory = new HessianProxyFactory();

HelloHessian helloHessian = (HelloHessian)factory.create(HelloHessian.class,url)

helloHessian.sayHello(new User(10, "tiger"))

2.1 通过HessianProxyFactory hessian代理工程类 创建代理类HessianProxy 具体代码如下:

public Object create(Class<?> api, URL url, ClassLoader loader) {

    ......

    InvocationHandler handler = new HessianProxy(url, this, api);

    return Proxy.newProxyInstance(loader, new Class[] { api, HessianRemoteObject.class }, handler);

}

2.2  HessianProxy 源码分析:

代理类HessianProxy 实现了InvocationHandler 接口 invoke(Object proxy, Method method, Object[] args)

2.2.1  获取调用的方法名以及方法参数类型

    String methodName = method.getName();

2.2.2  equals、hashCode、getHessianType、getHessianURL、toString 本地调用

  if(is1.equals("hashCode") && conn.length == 0) {

     return new Integer(this._url.hashCode());

  }

  if(is1.equals("getHessianType")) {

    return proxy.getClass().getInterfaces()[0].getName();

  }

2.2.3  重载支持 重载处理 mangleName = sayHello_User // 方法名_参数1_参数2_XX

2.2.4  执行远程调用(建立http链接,设置http header 信息, 序列化方法和参数,执行http请求)

//建立http链接

URLConnection conn = this._factory.openConnection(this._url);

//设置http header 信息

this.addRequestHeaders(conn);

//序列化并输出流

AbstractHessianOutput out = this._factory.getHessianOutput(conn.getOutputStream());

out .call(mangleName , args)  // out .call("sayHello",new Object[]{user})

out .flush();

call 方法封装hessian 版本信息、调用的方法名称、参数信息,格式如下

‘c‘ 一个调用方法编码的开始标识符

1和0 Hessian的版本号,各占1 byte

length 方法名称的长度,占2 byte

method_name 方法名称

[params] 参数序列化内容,此参数是可选的

‘z‘ 一个调用方法编码的结束标识符

eg:

User user = new User(10,"tiger");

FileOutputStream fos = new FileOutputStream("out.txt");

HessianOutput out = new HessianOutput(fos);

out.call("sayHello",new Object[]{user});

用BinaryViewer 工具打开 out.txt ,查看内容:此处为16进制数表示

63 01 00 6D 00 08 73 61 79 48 65 6C 6C 6F 4D 74

00 07 76 6F 2E 55 73 65 72 53 00 03 61 67 65 49

00 00 00 0A 53 00 04 6E 61 6D 65 53 00 05 74 69

67 65 72 7A 7A

63为‘c‘字符的编码

01 00为Hessian协议版本

6D为‘m‘字符编码

00 08 为函数名称的长度,因为‘sayHello’函数的长度为8

73 61 79 48 65 6C 6C 6F 即为‘sayHello‘函数名称的字符编码;

7A为‘z‘的字符编码

2.2.5  获取输入流并反序列

is = conn.getInputStream();

AbstractHessianInput in = _factory.getHessianInput(is);

Object value = in.readObject(method.getReturnType())

 

 

时间: 2024-12-28 00:47:29

hessian原理解析一(客户端分析)的相关文章

hessian原理解析三(序列化协议)

1.序列化 序列化:将数据结构或对象转换成二进制串的过程 反序列化:将在序列化过程中所生成的二进制串转换成数据结构或者对象的过程 目的:保存对象状态或用于网络传输 2.hessian 序列化协议 1.0 参考官方文档:http://hessian.caucho.com/doc/hessian-1.0-spec.xtp 9 primitive types boolean 32-bit int 64-bit long 64-bit double 64-bit date UTF8-encoded st

游戏外挂原理解析与制作 - [内存数值修改类 篇一]

本章旨在讲解外挂实现原理,未深入涉及至代码层面.希望能与对这方面感兴趣的朋友多多交流,毕竟理论是死的,套路是固定的,只有破解经验是花大量时间和心血积累的. 对于单机游戏而言,游戏中绝大部分的参数(比如血.蓝.能量亦或是金币)都存储在计算机的堆栈中,一些类似剧情进度的则加密后写入本地的自定义配置文件中: 对于页游.网游和手游,虽然服务器保存了大量的重要的参数,但由于客户端不可避免的需要进行大量的计算和资源的加载,本地内存种必定存有部分的临时变量,通过判断这些变量的变化规律和函数的破密寻到利于自身的

Android应用市场省流量更新(增量升级)原理解析

一.前言 最近在看热修复相关的框架,之前我们已经看过了阿里的Dexposed和AndFix这两个框架了,不了解的同学可以点击这里进行查看:Dexposed框架原理解析 和 AndFix热修复框架原理解析,然后还有最近很火的一个是腾讯的Tinker热修复框架,再看他的原理实现的时候,发现了他使用到了开源的文件差分工具bsdiff/bspatch,所以就单独用这篇文章来详细介绍一下这个工具,因为这个工具有一个很大的用途就是增量更新,也就是我们看到现在大部分的应用市场推出的省流量更新应用的效果: 看到

数据库水平切分的实现原理解析

数据库水平切分的实现原理解析---分库,分表,主从,集群,负载均衡器 随着互联网应用的广泛普及,海量数据的存储和访问成为了系统设计的瓶颈问题.对于一个大型的互联网应用,每天几十亿的PV无疑对数据库造成了相当高的负载.对于系统的稳定性和扩展性造成了极大的问题.通过数据切分来提高网站性能,横向扩展数据层已经成为架构研发人员首选的方式.水平切分数据库,可以降低单台机器的负载,同时最大限度的降低了了宕机造成的损失.通过负载均衡策略,有效的降低了单台机器的访问负载,降低了宕机的可能性:通过集群方案,解决了

Log4js原理解析

基于log4js 0.6.14版本 Log4js总共三篇博客 <Log4js原理解析>http://write.blog.csdn.net/postedit/42844085 <Log4js配置详解>http://blog.csdn.net/hfty290/article/details/42843737 <Log4js多进程陷阱与避免>http://blog.csdn.net/hfty290/article/details/42843303 一.概述 网络上有不少关于

【Java】Servlet 工作原理解析

Web 技术成为当今主流的互联网 Web 应用技术之一,而 Servlet 是 Java Web 技术的核心基础.因而掌握 Servlet 的工作原理是成为一名合格的 Java Web 技术开发人员的基本要求.本文将带你认识 Java Web 技术是如何基于 Servlet 工作,你将知道:以 Tomcat 为例了解 Servlet 容器是如何工作的?一个 Web 工程在 Servlet 容器中是如何启动的? Servlet 容器如何解析你在 web.xml 中定义的 Servlet ?用户的请

Jetty的工作原理解析以及与Tomcat的比较

Jetty 的基本架构 Jetty 目前的是一个比较被看好的 Servlet 引擎,它的架构比较简单,也是一个可扩展性和非常灵活的应用服务器,它有一个基本数据模型,这个数据模型就是 Handler,所有可以被扩展的组件都可以作为一个 Handler,添加到 Server 中,Jetty 就是帮你管理这些 Handler. Jetty 的基本架构 下图是 Jetty 的基本架构图,整个 Jetty 的核心组件由 Server 和 Connector 两个组件构成,整个 Server 组件是基于 H

android黑科技系列——应用市场省流量更新(增量升级)原理解析

一.前言 最近在看热修复相关的框架,之前我们已经看过了阿里的Dexposed和AndFix这两个框架了,不了解的同学可以点击这里进行查看:Dexposed框架原理解析 和 AndFix热修复框架原理解析,然后还有最近很火的一个是腾讯的Tinker热修复框架,再看他的原理实现的时候,发现了他使用到了开源的文件差分工具bsdiff/bspatch,所以就单独用这篇文章来详细介绍一下这个工具,因为这个工具有一个很大的用途就是增量更新,也就是我们看到现在大部分的应用市场推出的省流量更新应用的效果: 看到

(转)Apache和Nginx运行原理解析

Apache和Nginx运行原理解析 原文:https://www.server110.com/nginx/201402/6543.html Web服务器 Web服务器也称为WWW(WORLD WIDE WEB)服务器,主要功能是提供网上信息浏览服务. 应用层使用HTTP协议. HTML文档格式. 浏览器统一资源定位器(URL). Web服务器常常以B/S(Browser/Server)方式提供服务.浏览器和服务器的交互方式如下: GET /index.php HTTP/1.1 +-------