概要介绍
在我们的项目中,特别是在J2EE分布式系统中,我们经常需要选择某种远程服务协议,在分布的进程间来进行交互和系统数据调用。目前可供选择的协议 EJB、 基于SOAP的Web Service 这些都是重量级的,也有像RMI、Socket这样的比较原始的。但是Hessian/burlap/HttpInvoker是比较优秀的,也曾在多个项目和工程中使用,性能比较好,也比较稳定,如阿里巴巴(北京-杭州)和淘宝内部(杭州-北京)的对象交互都是采用这种方法。实践证明,传输同样的对象,Hessian协议传输的数据量比SOAP协议低一个数量级。因此在复杂网络环境下的分布式应用使用Hessian协议可以获得更好的性能和可靠性。
Hessian和Burlap是由Caucho公司提供的方案,如果把Hessian/Burlap与Spring结合使用,设计者将感到无比的方便。Spring目前提供了对RMI、HttpInvoker、Hessian、Burlap及WebService等Remoting技术的集成。而Spring又是目前业务逻辑层常用的框架,因此采用在spring中使用Hessian来传输二进制流是符合开发效果和成本的。Hessian/Burlap/HttpInvoker将参数和返回值通过Java的序列化机制进行编组和反编组,它具有RMI的支持所有可序列化对象的优点。是使用Http协议传输二进制流的,Hessian的binary协议和web service常用的SOAP协议类似,也是将协议报文封装在HTTP封包中,通过HTTP信道传输的。因此Hessian协议具有与SOAP协议同样的优点,即传输不受防火墙的限制(防火墙通常不限制HTTP信道)。Hessian和Burlap解决了RMI头疼的防火墙问题。并且它们都是非常轻量级的,足以在内存或空间受限制的环境下使用,例如applet和无线设备。下面看一下几种基于HTTP协议的RPC性能比较(来源网上):
l Axis ---------->> Total time: 87359 ms, Avg time: 8.7359 ms
l Burlap --------->> Total time: 5784 ms, Avg time: 0.5784 ms
l Hessian -------->> Total time: 5084 ms, Avg time: 0.5084 ms
l REST ---------->> Total time: 7983 ms, Avg time: 0.7983 ms
l AxisUsingWSDL2Java ->> Total time: 113234 ms, Avg time: 11.3234 ms
从中可以看出,在相同的环境下Hessian具有非凡的性能和表现。
下面我们来看一下一个真实的Demo和开发步骤。
开发环境:
Myeclipse6.5和JDK1.6、Spring 2.5和Resin3.13
一.Server端的开发及服务发布
1.创建工程及代码类
建立一个web project,我取名为APIServer,创建包及包名和我们要开放服务器的接口及实现类。
接口很简单,就是返回用户对象的List。(假如这个server要提供用户信息给远程系统或者服务)
public interface DemoService {
List<SyncObject> getUsers();
}
上面接口的实现:
public class DemoServiceImpl implements DemoService {
public List<SyncObject> getUsers(){
//可以从数据库查询或者文件获取所需的数据进行封装
//此处为了测试写死一些数据。
List<SyncObject> lis = new ArrayList();
SyncObject user1 = new SyncObject();
SyncObject user2 = new SyncObject();
user1.setUsername("username1 from server");
user1.setPassword("password1 from server");
user2.setUsername("username2 from server");
user2.setPassword("password2 from server");
lis.add(user1);
lis.add(user2);
return lis;
}
}
2.配置文件信息
配置web.xml
<servlet>
<servlet-name>dispatcher</servlet-name>
<servlet-class>
<!-- web container把所有的请求都交给DispatcherServlet -->
org.springframework.web.servlet.DispatcherServlet
</servlet-class>
<load-on-startup>0</load-on-startup>
</servlet>
<servlet-mapping>
<!-- 所有匹配/remoting/*的请求被DispatcherServlet 截获-->
<servlet-name>dispatcher</servlet-name>
<url-pattern>/remote/*</url-pattern>
</servlet-mapping>
<context-param>
<param-name>log4jConfigLocation</param-name>
<param-value>/WEB-INF/log4j.properties</param-value>
</context-param>
<listener>
<listener-class>
com.sitech.vasd.server.servlet.SystemConfigListener
</listener-class>
</listener>
配置dispatcher-servlet.xml
<!-- 业务类 -->
<bean id="demoService" class="com.demo.api.server.DemoServiceImpl" />
<!-- 远程服务 -->
<!-- 使用HessianServiceExporter 将普通bean导出成Hessian服务-->
<bean name="/HessianService" class="org.springframework.remoting.caucho.HessianServiceExporter">
<!-- 需要导出的目标bean-->
<property name="service" ref="demoService" />
<!-- Hessian服务的接口-->
<!-- Hessian服务的接口-->
<property name="serviceInterface" value="com.demo.api.server.DemoService" />
</bean>
在该配置下,Hessian服务的url是
http://IP:port/**/remote/HessianService,其中**是该应用的url。Spring使用DispatcherServlet拦截到匹配/remoting/*的请求,然后将该请求转发到对应的bean,该bean在dispatcher-servlet.xml文件中以HessianServiceExporter定义。
二.客户端
同样我们创建工程和代码
Web project&code
public interface DemoService {
List getUsers();
}
下面注意了:
创建可序列化对象,就是Server端的那个syncobject,而且要implements Serializable。这个也是我们要远程对象调用的domain类。
public class SyncObject implements Serializable {
/**
* Comment for <code>serialVersionUID</code>
*/
private static final long serialVersionUID = 1L;
private String username="";
private String password="";
public String getUsername() {
return username;
}
public void setUsername(String username) {
this.username = username;
}
public String getPassword() {
return password;
}
public void setPassword(String password) {
this.password = password;
}
}
配置文件config.xml
<bean id="hessianService" class="org.springframework.remoting.caucho.HessianProxyFactoryBean">
<!--服务器端发布的地址URL-->
<property name="serviceUrl" value="http://ip:port/**/remote/HessianService" />
<!--本地接口à
<property name="serviceInterface" value="com.demo.api.client.DemoService"/>
</bean>
最后我们写个测试方法:
public static void main(String[] args) {
final ApplicationContext context = new ClassPathXmlApplicationContext("config.xml");
DemoService service = (DemoService)context.getBean("hessianService");
try {
List<SyncObject> userlist= service.getUsers();
for (SyncObject user : userlist) {
System.out.println(user.getUsername());
}
}
catch(RemoteException e) {
e.printStackTrace();
}
}
Run一把试试,(确保服务器端runing状态哦)
下面是成功后的信息,结果是从服务器端取到了2个用户的username。
2008-12-31 10:31:15 org.springframework.core.CollectionFactory <clinit>
信息: JDK 1.4+ collections available
2008-12-31 10:31:16 org.springframework.beans.factory.xml.XmlBeanDefinitionReader loadBeanDefinitions
信息: Loading XML bean definitions from class path resource [config.xml]
2008-12-31 10:31:16 org.springframework.context.support.AbstractRefreshableApplicationContext refreshBeanFactory
信息: Bean factory for application context [org.springframework.context.support.ClassPathXmlApplicationContext;hashCode=11988197]: org.springframework.beans.factory.support.DefaultListableBeanFactory defining beans [hessianService]; root of BeanFactory hierarchy
2008-12-31 10:31:16 org.springframework.context.support.AbstractApplicationContext refresh
信息: 1 beans defined in application context [org.springframework.context.support.ClassPathXmlApplicationContext;hashCode=11988197]
2008-12-31 10:31:16 org.springframework.context.support.AbstractApplicationContext initMessageSource
信息: Unable to locate MessageSource with name 'messageSource': using default [org.springframework.context.support.DelegatingMessageSource@b8f82d]
2008-12-31 10:31:16 org.springframework.context.support.AbstractApplicationContext initApplicationEventMulticaster
信息: Unable to locate ApplicationEventMulticaster with name 'applicationEventMulticaster': using default [org.springframework.context.event.SimpleApplicationEventMulticaster@b1b4c3]
2008-12-31 10:31:16 org.springframework.beans.factory.support.DefaultListableBeanFactory preInstantiateSingletons
信息: Pre-instantiating singletons in factory [org.springframework.beans.factory.support.DefaultListableBeanFactory defining beans [hessianService]; root of BeanFactory hierarchy]
2008-12-31 10:31:16 org.springframework.aop.framework.DefaultAopProxyFactory <clinit>
信息: CGLIB2 not available: proxyTargetClass feature disabled
username1 from server
username2 from server