`
merry爱编程
  • 浏览: 9560 次
  • 性别: Icon_minigender_1
  • 来自: 哈尔滨
社区版块
存档分类

Hessian:轻量级的remoting onhttp工具介绍及基于Spring2的完整实例

    博客分类:
  • java
阅读更多
声明:一下文章摘自Ciber's Java Blog 地址:http://www.blogjava.net/behone/archive/2011/07/11/354128.html


Hessian:轻量级的remoting onhttp工具介绍及基于Spring2的完整实例
一、Hessian简介(摘自百度百科)

Hessian是一个轻量级的remoting onhttp工具,使用简单的方法提供了RMI的功能. 相比WebService,Hessian更简单、快捷。采用的是二进制RPC协议,因为采用的是二进制协议,所以它很适合于发送二进制数据

二、Hessian开发要点
1、JAVA服务器端必须具备以下几点:
(1)包含Hessian的jar包
(2)设计一个接口,用来给客户端调用
(3)该接口的实现类
(4)配置web.xml中的servlet
(5)对象必须实现Serializable 接口
(6)对于复杂对象可以使用Map的方法传递
(7)不支持JDK1.6+weblogic11g+Spring
-2、客户端必须具备以下几点:
(1)包含Hessian的jar包。
(2)具有和服务器端结构一样的接口和交互对象。
(3)利用HessianProxyFactory调用远程接口。

三、基于Spring2.0的Hessian开发实例
1、环境:
(1)服务端:JDK1.4+weblogic8.1
(2)客户端:JDK1.6+weblogic11g
2、相关JAR包:
(1)服务端:spring.jar、hessian-2.1.12.jar、commons-logging-1.0.4.jar
(2)客户端:spring.jar、hessian-2.1.12.jar、commons-logging-1.0.4.jar
四、服务端的实现
1、相关model(com.govdo.model)
(1)ServiceRequest.java 客户端请求服务类,用于接受客户端请求数据

1package com.govdo.model;
2
3import java.io.Serializable;
4import java.util.HashMap;
5import java.util.Map;
6
7public class ServiceRequest implements Serializable {
8
9    private static final long serialVersionUID = 1L;
10    private static final String REQUESTED_SERVICE_ID = "REQUESTED_SERVICE_ID";
11    private static final String CURRENT_REQUEST_OBJECT = "CURRENT_REQUEST_OBJECT";
12    private Map parameters;
13    public ServiceRequest() {
14        this.parameters = new HashMap();
15    }
16    public ServiceRequest(Map parameters) {
17        if (parameters != null) {
18            this.parameters = parameters;
19        }
20        this.parameters = new HashMap();
21    }
22    public Object getParameter(String parameterKey) {
23        return this.parameters.get(parameterKey);
24    }
25    public void setParameter(String parameterKey, Object parameterValue) {
26        this.parameters.put(parameterKey, parameterValue);
27    }
28    public String getRequestedServiceID() {
29        return (String)this.parameters.get(REQUESTED_SERVICE_ID);
30    }
31    public void setRequestedServiceID(String serviceID) {
32        this.parameters.put(REQUESTED_SERVICE_ID, serviceID);
33    }
34    public Object getCurrentRequestObject() {
35        return this.parameters.get(CURRENT_REQUEST_OBJECT);
36    }
37    public void setCurrentRequestObject(Object object) {
38        this.parameters.put(CURRENT_REQUEST_OBJECT, object);
39    }
40    public Map getParameters() {
41        return parameters;
42    }
43    public void setParameters(Map parameters) {
44        this.parameters =  parameters;
45    }
46}
47
(2)ServiceResponse.java 客户端响应服务类,用于返回客户端响应数据
package com.govdo.model;

import java.io.Serializable;
import java.util.Map;

public class ServiceResponse implements Serializable {

    private static final long serialVersionUID = 1L;
    public static final String SERVICE_RESPONSE_RESULT = "SERVICE_RESPONSE_RESULT";
    public static final String BUSINESS_SUCCESS = "0";
    public static final String BUSINESS_FAILURE = "1";

    private Map model = null;

    public ServiceResponse() {}
    public ServiceResponse(Map model) {
        this.model = model;
    }
    public Map getModel() {
        return model;
    }

    public void setModel(Map model) {
        this.model = model;
    }

}
(3)QueryCityInfoIn.java 城市信息客户端请求对象,实现Serializable,源码略
(4)QueryCityInfoOut.java 城市信息客户端响应对象,实现Serializable,源码略
2、用于客户端调用的公共接口及实现
(1)Action.java 暴露给客户端的抽象公共接口
package com.govdo.action;

import com.govdo.model.ServiceRequest;
import com.govdo.model.ServiceResponse;

public abstract interface Action {

    public abstract ServiceResponse perform(ServiceRequest request) throws Exception;
}
(2)AbstractAction.java 实现Action、BeanFactoryAware接口的抽象类
package com.govdo.action;

import org.springframework.beans.BeansException;
import org.springframework.beans.factory.BeanFactory;
import org.springframework.beans.factory.BeanFactoryAware;

import com.govdo.model.ServiceRequest;
import com.govdo.model.ServiceResponse;

public abstract class AbstractAction implements Action, BeanFactoryAware {

    protected BeanFactory context;
    public void setBeanFactory(BeanFactory context) throws BeansException {
        this.context = context;
    }

    public abstract ServiceResponse perform(ServiceRequest request)
            throws Exception;
}
(3)RemotingAction 继承AbstractAction的公共类,用于远程实现接口Action,其中serviceID是真正实现功能的bean,这个bean必须实现perform方法
package com.govdo.action;

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;

import com.govdo.model.ServiceRequest;
import com.govdo.model.ServiceResponse;
import com.govdo.util.StringUtils;

public class RemotingAction extends AbstractAction {

    private Log logger = LogFactory.getLog(getClass());
    public ServiceResponse perform(ServiceRequest request)
            throws Exception {
        String serviceID = (String)request.getRequestedServiceID();
        if (StringUtils.isBlank(serviceID)) {
            logger.error("service id isn't allowed to be null!");
            throw new IllegalArgumentException("service id isn't allowed to be null!");
        }
        Action action = (Action)context.getBean(serviceID);
        return action.perform(request);
    }
}
3、服务端的相关配置
(1)remoting-servlet.xml 在WEB-INF下添加这个xml文件,其中/myHessian是暴露给客户端的访问路经,Action是调用接口,remotingAction是公共实现类
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE beans PUBLIC "-//SPRING//DTD BEAN//EN"
    "http://www.springframework.org/dtd/spring-beans.dtd">
<beans>
    <bean id="remotingAction" class="com.govdo.action.RemotingAction">
    </bean>
    <bean name="/myHessian"
        class="org.springframework.remoting.caucho.HessianServiceExporter">
        <property name="service" ref="remotingAction" />
        <property name="serviceInterface" value="com.govdo.action.Action" />
    </bean>
</beans>
(2)web.xml中增加相应的servlet配置,这里采用Spring的org.springframework.web.servlet.DispatcherServlet
<?xml version="1.0" encoding="UTF-8"?>
<web-app version="2.4" xmlns="http://java.sun.com/xml/ns/j2ee"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee
    http://java.sun.com/xml/ns/j2ee/web-app_2_4.xsd">

    <welcome-file-list>
        <welcome-file>index.jsp</welcome-file>
    </welcome-file-list>

    <servlet>
        <servlet-name>remoting</servlet-name>
        <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
        <init-param>
            <param-name>contextConfigLocation</param-name>
            <param-value>
            ,/WEB-INF/remoting-servlet.xml
            </param-value>
        </init-param>
        <load-on-startup>1</load-on-startup>
    </servlet>

    <servlet-mapping>
        <servlet-name>remoting</servlet-name>
        <url-pattern>/remoting/*</url-pattern>
    </servlet-mapping>

</web-app>
4、增加具体实现功能的bean
(1)HessianTestImpl.java具体实现类,继承AbstractAction相当于实现了Action接口,这里对请求数据做了简单处理,对城市名称后加了(by remoting),然后返回客户端,注意要实现perform方法
package com.govdo.test.action;

import java.util.HashMap;
import java.util.Map;

import org.springframework.beans.BeanUtils;

import com.govdo.action.AbstractAction;
import com.govdo.model.QueryCityInfoIn;
import com.govdo.model.QueryCityInfoOut;
import com.govdo.model.ServiceRequest;
import com.govdo.model.ServiceResponse;

public class HessianTestImpl extends AbstractAction {

    public ServiceResponse perform(ServiceRequest request) throws Exception {

        Map resultMap = new HashMap();
        QueryCityInfoIn in = (QueryCityInfoIn)request.getCurrentRequestObject();
        QueryCityInfoOut out = new QueryCityInfoOut();
        BeanUtils.copyProperties(in, out);
        out.setCityName(out.getCityName()+"(by remoting)");
        resultMap.put(ServiceResponse.BUSINESS_SUCCESS, out);
        Map model = new HashMap();
        model.put(ServiceResponse.SERVICE_RESPONSE_RESULT, resultMap);
        return new ServiceResponse(model);
    }
}
(2)在Spring提供bean服务的XML配置文件中增加一个Bean(不妨就加在remoting-servlet.xml中,也可以另写一个xml,但必须在web.xml中加载),后面客户端实现会用到这个bean。
<bean id="hessianTest" class="com.govdo.test.action.HessianTestImpl">
    </bean>
到这里服务端实现已全部完成,打包部署完工。这里假设部署实例上下文路径为:http://127.0.0.1:7008/webRoot/

五、客户端实现
1、准备工作
(1)相关model,服务端的4个对象都需要加到客户端来
(2)客户端调用的接口,Action接口也要加到客户端来
2、远程调用相关配置
(1)remoting-client.xml 配置远程调用接口和URL,回顾上章3(1)节,/myHessian是在remoting-servlet.xml 中配置的远程访问服务路径,所以这里URL为http://127.0.0.1:7008/webRoot/remoting/myHessian,而接口则只需配Action这个公共接口即可
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE beans PUBLIC "-//SPRING//DTD BEAN//EN"
    "http://www.springframework.org/dtd/spring-beans.dtd">
<beans>
    <bean id="hessianTestClient"
        class="org.springframework.remoting.caucho.HessianProxyFactoryBean">
        <property name="serviceUrl"
            value="http://127.0.0.1:7008/webRoot/remoting/myHessian" />
        <property name="serviceInterface" value="com.govdo.action.Action" />
    </bean>
</beans>
(2)web.xml中加载remoting-client.xml (如果是在main函数中测试,无需配置web.xml)
3、客户端调用测试类TestHessianTest.java
package com.govdo.test.action;

import java.util.Map;

import org.springframework.context.support.ClassPathXmlApplicationContext;

import com.govdo.action.Action;
import com.govdo.model.QueryCityInfoIn;
import com.govdo.model.QueryCityInfoOut;
import com.govdo.model.ServiceRequest;
import com.govdo.model.ServiceResponse;

public class TestHessianTest {

    public static void main(String[] args) {
        ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("remoting-client.xml");
        try {
            QueryCityInfoIn in = new QueryCityInfoIn();
            in.setCityId("Shenzhen");
            in.setCityName("深圳");
            Action ac = (Action)context.getBean("hessianTestClient");
            ServiceRequest serviceRequest = new ServiceRequest();
            serviceRequest.setRequestedServiceID("hessianTest");
            serviceRequest.setCurrentRequestObject(in);
            ServiceResponse serviceResponse = ac.perform(serviceRequest);
            Map model = (Map)serviceResponse.getModel();
            if (model != null) {
                Map resultList = (Map)(model.get(ServiceResponse.SERVICE_RESPONSE_RESULT));
                QueryCityInfoOut out = (QueryCityInfoOut)(resultList.get(ServiceResponse.BUSINESS_SUCCESS));
                System.out.println("CityId:"+out.getCityId()+";CityName:"+out.getCityName());
            }
        } catch (Exception e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }

    }
}
运行结果:CityId:Shenzhen;CityName:深圳(by remoting)
这个(by remoting)就是通过服务端程序加的,请回顾上章4(1)节,至此客户端调用实现结束。

六、小结
对于服务端的这种实现方式,有以下优缺点
1、优点
(1)配置简单扩展性好,服务端只需一次配置,客户端即可调用多个实现不同功能的实例,如文章例中的hessianTest是通过serviceRequest.setRequestedServiceID("hessianTest")去获取,也就是说服务端的其他实现了Action接口的bean都可以通过这个方法远程获取并调用
2、缺点
(1)目前经本人测试,hessian的多个版本(包括最老的最新的)都不支持JDK1.6+weblogic11g的Spring配置,而weblogic11g需要依赖JDK1.6以上版本,weblogic8.1不支持1.4以上版本,所以服务端想用泛型就不可能了,本人对泛型情有独钟,本打算出一个Spring+hessian+ibatis的介绍,看来只能分开讲了。
分享到:
评论

相关推荐

    外部接口调用 使用spring4+hessian4实例(二)

    Hessian是一种轻量级的二进制Remoting协议,由Caucho公司开发。它允许开发者通过HTTP协议快速、高效地进行远程方法调用(RPC)。Hessian提供了一种序列化机制,将Java对象转化为二进制流,使得在网络间传输时更节省...

    外部接口调用 使用spring4+hessian4实例

    Spring框架是Java领域广泛使用的轻量级框架,它提供了丰富的功能,包括依赖注入、面向切面编程以及服务集成等。而Hessian是一种二进制RPC协议,它使得远程方法调用变得更加高效,尤其适合于处理简单的HTTP通信。 ...

    spring、hessian通过tomcat的简单环境应用源代码

    在实际开发中,这种集成方案常用于分布式系统,特别是在需要快速、轻量级的跨网络服务通信时,Hessian提供的二进制序列化比基于XML的方案如SOAP更为高效。而Spring则负责管理和协调这些服务,提供了一种声明式、面向...

    spring远程调用简单实例

    在本实例中,我们关注的是HTTP Invoker,这是一种基于HTTP的轻量级远程调用解决方案。HTTP Invoker不需要额外的二进制协议支持,而是利用Java序列化机制,将方法调用和参数以HTTP请求的形式发送到远程服务器,然后...

    hessin+spring实例

    Hessian是一种轻量级的远程过程调用(RPC)协议,相比传统的WebService技术,Hessian提供了更简单且高效的方式来实现分布式应用间的通信。Hessian的核心优势在于其采用了二进制RPC协议(Binary RPC),这种协议相比于...

    spring使用注解暴露remoting服务

    Hessian的优点在于其轻量级和高效的特性,可以快速地实现远程方法调用,尤其适合Java与Java之间的通信。在Spring中,可以使用`HessianServiceExporter`类来暴露一个服务为Hessian服务,通过简单的配置,就可以让其他...

    SpringCloud我服务基础教程

    - 轻量级的remoting on HTTP工具,基于HTTP协议,采用二进制编解码。 - **Thrift**: - 一种可伸缩的跨语言服务的软件框架,支持多种语言,通过定义描述文件自动生成客户端和服务端代码。 - **SpringCloud**: - ...

    Spring2.5jar所有开发包及完整文档及项开发实例.pdf

    Spring框架是Java应用程序开发中的一个核心库,它提供了一个丰富的基础设施,用于简化企业级应用的开发。Spring 2.5版本包含了多个jar包,每个jar包都有特定的用途,允许开发者根据需求选择合适的依赖。 首先,`...

    Spring(RMI)实例

    Spring框架是Java开发中广泛应用的轻量级框架,它提供了丰富的功能来简化应用程序的构建,包括依赖注入、AOP(面向切面编程)以及多种集成技术。RMI(Remote Method Invocation,远程方法调用)是Java平台上的一个...

    使用httpInvoker提供远程服务

    与Hessian等轻量级通信协议不同,HttpInvoker依赖于Java的序列化机制来处理方法的参数和返回值。 **14.7.1 输出业务对象** 使用HttpInvoker服务出口商(HttpInvokerServiceExporter)可以将普通的JavaBean实例转换...

    Dubbo 项目结构解析1

    11. **hessian-lite**:轻量级Hessian库,用于序列化和反序列化,提升服务调用效率。 12. **dubbo-demo**:示例模块,包含了Dubbo基本使用方法的示例代码,帮助初学者快速入门。 13. **dubbo-test**:测试模块,...

    Java远程技术.pdf

    - JBoss-Remoting:一个轻量级的远程通信框架。 - Spring Remoting:Spring框架提供的远程方法调用支持。 - ActiveMQ:一个开源的消息代理和面向消息的中间件。 - Mule ESB:一个开源的企业服务总线。 Java远程...

    J2EE开发全程实录PDF J2EE开发全程实录PDF

    - **Hessian使用演示**:Hessian是一种轻量级的远程过程调用协议,这里通过示例展示了如何使用Hessian实现远程调用。 - **几种Remoting实现的比较**:对比分析了多种远程调用技术的特点和适用场景。 - **改造...

    spring-remote-service-example:一个示例项目,用于展示如何使用 JSON 反序列化通过 HTTP 在隔离的 Spring 容器之间进行(远程)服务调用

    2. **JSON反序列化**:JSON(JavaScript Object Notation)是一种轻量级的数据交换格式,易于人阅读和编写,同时也易于机器解析和生成。在Java中,Jackson库被广泛用于JSON的序列化和反序列化。通过Jackson,我们...

    alibaba-dubbo-dubbo-2.5.7-0-ge2d63ad.tar.gz

    阿里巴巴的Dubbo是一款高性能、轻量级的开源Java RPC框架,它提供了服务发现、服务治理、流量控制等一系列完整的服务解决方案。本压缩包"alibaba-dubbo-dubbo-2.5.7-0-ge2d63ad.tar.gz"包含了Dubbo的2.5.7版本的源...

    dubbo源码解析2

    Dubbo是一个高性能、轻量级的开源Java RPC框架,它提供了服务自动注册与发现、透明化的远程方法调用、智能负载均衡等功能。Dubbo的设计理念是“为服务而生”,旨在简化微服务架构下的服务管理和调用。 #### 三、...

Global site tag (gtag.js) - Google Analytics