`
hanqunfeng
  • 浏览: 1527148 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

Spring HTTP invoker简介

阅读更多

Spring HTTP invoker 简介

Spring HTTP invoker spring 框架中的一个远程调用模型,执行基于 HTTP 的远程调用(意味着可以通过防火墙),并使用 java 的序列化机制在网络间传递对象。客户端可以很轻松的像调用本地对象一样调用远程服务器上的对象,这有点类似于 webservice ,但又不同于 webservice ,区别如下:

webservice

HTTP invoker

跨平台,跨语言

只支持 java 语言

支持 SOAP ,提供 wsdl

不支持

结构庞大,依赖特定的 webservice 实现,如 xfire

结构简单,只依赖于 spring 框架本身

项目中使用哪种远程调用机制取决于项目本身的要求。

² HTTP invoker 服务模式

HTTP invoker 服务模式

 

说明:

1. 服务器端:通过 HTTP invoker 服务将服务接口的某个实现类提供为远程服务

2. 客户端:通过 HTTP invoker 代理向服务器端发送请求,远程调用服务接口的方法

3. 服务器端与客户端通信的数据需要序列化

 

 

配置服务器端和客户端的步骤

配置服务器端

1. 添加 springJAR 文件

建议使用 spring2+.jar 版本

2. 创建服务接口

3. 创建服务接口的具体实现类

4. 公开服务

 

配置客户端

1. 添加 springJAR 文件

建议使用 spring2+.jar 版本

2. 创建服务接口

3. 访问服务

 

实例讲解

服务器端

1. 服务接口: UcService.java

它提供两项服务,查询用户信息和记录日志,如下:

public interface UcService {

public UserInfo getUserInfobyName(String userName);

public int recordLog(String username, String point, String operate, String desc);

}

说明:举这个列子是因为其比较有代表性,它将展示普通数据类型( int,long 等)和复杂数据类型( DTO 等)的远程调用方式。 UserInfo 是一个普通的 DTO ,代码如下:

public class UserInfo implements Serializable {

private static final long serialVersionUID = -6970967506712260305L;

/** 用户名 */

private String userName ;

/** 电子邮箱 */

private String email ;

/** 注册日期 */

private Date registDate ;

 

public String getUserName() {

return userName ;

}

public void setUserName(String userName) {

this . userName = userName;

}

public String getEmail() {

return email ;

}

public void setEmail(String email) {

this . email = email;

}

public Date getRegistDate() {

return registDate ;

}

public void setRegistDate(Date registDate) {

this . registDate = registDate;

}

}

注意:因为是在网络间传输对象,所以需要将 UserInfo 实现 Serializable 接口,并指定一个 serialVersionUID (任意值即可,同时客户端也要有这个类,否则在客户端接收对象时会因为 serialVersionUID 不匹配而出现异常)

回到UcService.java ,它提供了两个服务(在这里一个方法代表一个服务功能),我们需要具体的实现类来实现真正的服务

 

2. 实现类是 UCServiceImpl.java

public class UCServiceImpl implements UcService {

private static Logger pointrecordlog = Logger.getLogger ( "pointrecordlog" );

private static Logger logger = Logger.getLogger (UCServiceImpl. class );

private UcFacade ucFacade ;

public void setUcFacade(UcFacade ucFacade) {

this . ucFacade = ucFacade;

}

public UserInfo getUserInfobyName(String userName) {

UserInfo user = null ;

try {

user = ucFacade .getUserInfoDetail(userName);

logger .debug( "get userinfo success by username:" + userName);

} catch (Throwable t) {

logger .error( "get userinfo fail by username:" + userName, t);

}

return user;

}

public int recordLog(String username, String point, String operate, String desc) {

int result = 0;

try {

pointrecordlog .info(username + " - " + point + " - " + operate + " - " + desc);

} catch (Throwable t) {

result = -1;

logger .error(t);

}

return result;

}

}

说明: ucFacade 是通过 spring 注入的一个数据查询类,因为它与 http invoker 没有直接关系,所以不进行介绍。

3. 公开服务 UcService.java

² WEB-INF/application-context.xml :将接口声明为 HTTP invoker 服务

< bean id = "httpService"

class = "org.springframework.remoting.httpinvoker.HttpInvokerServiceExporter" >

< property name = "service" >

< ref bean = "ucService" />

</ property >

< property name = "serviceInterface"

value = "com.netqin.baike.service.UcService" >

</ property >

</ bean >

< bean id = "ucService" class = "com.netqin.baike.service.impl.UCServiceImpl" />

说明: HttpInvokerServiceExporter 实际上是一个 spring mvc 控制器,它处理客户端的请求并调用服务实现。

² WEB-INF/service-servlet.xml HttpInvokerServiceExporter 实际上是一个 spring mvc 控制器,所以需要为其 提供 spring URL 处理器,这里我们使用 SimpleUrlHandlerMapping

< bean

class = "org.springframework.web.servlet.handler.SimpleUrlHandlerMapping" >

< property name = "mappings" >

< props >

< prop key = "/httpService" > httpService </ prop >

</ props >

</ property >

</ bean >

² WEB-INF/web.xml :配置 spring 监听及 DispatcherServlet

< context-param >

< param-name > contextConfigLocation </ param-name >

< param-value >

/WEB-INF/application-context.xml

</ param-value >

</ context-param >

 

< listener >

< listener-class >

org.springframework.web.context.ContextLoaderListener

</ listener-class >

</ listener >

 

< servlet >

< servlet-name > service</ servlet-name >

< servlet-class >

            org.springframework.web.servlet.DispatcherServlet

        </ servlet-class >

        < load-on-startup > 1 </ load-on-startup >

    </ servlet >

 

    < servlet-mapping >

        < servlet-name > service </ servlet-name >

        < url-pattern > /service/* </ url-pattern >

    </ servlet-mapping >

说明:不了解为什么这么配置的可以去看看 spring mvc 方面的资料。

 

好了,经过以上配置,一个基于 spring HTTP invoker 的远程服务就完成了,服务的地址为:

http://${serviceName}:${port}/${contextPath}/service/httpService

 

客户端

1.        创建服务接口及网络间传输的 DTO

为了方便,可以将服务器端创建好的的 UcService.java UserInfo.java 拷贝到客户端 , 或打个 jar 包放到 lib 下。

2.        配置访问服务

²       WEB-INF/application-context.xml :如果项目中已经存在 spring 配置文件,则不需要创建该文件,需要配置 HTTP invoker 的代理

< bean id = "httpService"

class = "org.springframework.remoting.httpinvoker.HttpInvokerProxyFactoryBean" >

        < property name = "serviceUrl" >

            < value > http://${serviceName}:${port}/${contextPath}/service/httpService

</ value >

        </ property >

        < property name = "serviceInterface"

            value = "com.netqin.baike.service.UcService" >

        </ property >

</ bean >

说明:客户端使用 HttpInvokerProxyFactoryBean 代理客户端向服务器端发送请求,请求接口为 UcService 的服务

注意:需要修改 serviceUrl 为实际的服务器地址

²         WEB-INF/web.xml :配置 spring 监听

如果项目没有 spring 环境,则需要在 web.xml 中加入对 spring 的支持

< context-param >

        < param-name > contextConfigLocation </ param-name >

        < param-value >

            /WEB-INF/application-context.xml

        </ param-value >

    </ context-param >

 

    < listener >

        < listener-class >

            org.springframework.web.context.ContextLoaderListener

        </ listener-class >

</ listener >

 

3.        访问服务方法

u         读取 spring 上下文,以远程调用 getUserInfobyName 方法为例

²         jsp,servlet,action 等等文件中

UcService service = (UcService) WebApplicationContextUtils

        .getRequiredWebApplicationContext(

            request.getSession().getServletContext()).getBean(

            "httpService" );

UserInfo user = service .getUserInfobyName( "hanqunfeng" );

 

²         如果不想配置 spring 运行环境,可以使用如下方式:

ApplicationContext applicationContext

= new FileSystemXmlApplicationContext( "classpath:application-context.xml" );

service = (UcService) applicationContext.getBean( "httpService" );

u         依赖注入,远程调用 recordLog 方法为例

²         WEB-INF/application-context.xml 中加入如下配置:

< bean id = "abc" class = "com.netqin.test.abc" >

        < property name = "service" >

            < ref bean = "httpService" />

        </ property >

</ bean >

²         com.netqin.test.abc 中加入对 service set 方法:

private UcService service ;

    public void setService(UcService service){

        this . service = service;

    }

 

    public String recordUserLog(String username,String point,String operate,String desc){

        String result = service .recordLog(username, point, operate, desc);

        return result;

}

 

 

 

关于服务器端配置的补充说明:

 有一个误区:有些关于springMVC的书 上说,如果没有明确声明一个处理适配器,默认会使用 org.springframework.web.servlet.mvc.SimpleControllerHandlerAdapter,这个适配器 专门负责处理所有实现了

org.springframework.web.servlet.mvc.Controller 接口的处理器,我就是受其影响,认为 org.springframework.remoting.httpinvoker.HttpInvokerServiceExporter实现的是 org.springframework.web.HttpRequestHandler接口,所以按理说应该使用的处理适配器是 org.springframework.web.servlet.mvc.HttpRequestHandlerAdapter,但实际上并不会出现异 常。

其实,原因是因为spring默认会使用四个处理适配器(参看DispatcherServlet.properties,spring2.5,spring2.0只默认三个,2.5增加注解方式):

org.springframework.web.servlet.mvc.HttpRequestHandlerAdapter,/
 org.springframework.web.servlet.mvc.SimpleControllerHandlerAdapter,/
 org.springframework.web.servlet.mvc.throwaway.ThrowawayControllerHandlerAdapter,/
 org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter

 关于DispatcherServlet.properties的详细信息可以参看:

http://blog.csdn.net/hanqunfeng/archive/2010/01/08/5161319.aspx

但是,如果明确声明了其它的处理适配器,比如 org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter, 等等,则默认规则则会覆盖,需要明确声明 org.springframework.web.servlet.mvc.HttpRequestHandlerAdapter这个处理适配器,否则系 统会抛异常:

javax.servlet.ServletException: No adapter for handler [org.springframework.remoting.httpinvoker.HttpInvokerServiceExporter@179bd14]: Does your handler implement a supported interface like Controller?

所以,建议在使用spring invoker时,最好明确声明org.springframework.web.servlet.mvc.HttpRequestHandlerAdapter这个处理适配器

 

分享到:
评论

相关推荐

    Spring HttpInvoker介绍

    公司内部讲义,比较了SOA,RMI和Spring HttpInvoker。并介绍了Spring HttpInvoker的基本使用方法。

    Spring HttpInvoker

    Spring HttpInvoker,是一套基于Maven+Spring+SpringMVC+MyBatis框架,还包含了Invoker的客户端及服务器端的demo实例

    Spring 实现远程访问详解——httpinvoker

    Spring httpInvoker使用标准java序列化机制,通过Http暴露业务服务。如果你的参数和返回值是比较复杂的,通过httpInvoker有巨大的优势。 1. 远程访问流程 1) 服务端定义服务接口 2) 服务端实现服务接口 3) 暴露服务...

    Spring HttpInvoker 初学实例

    org.springframework.remoting.httpinvoker最基本的实现例子,这是3个eclipse的JavaEE工程,全部导入即可,能运行。 初学可以轻松了解Spring HttpInvoker 的结构和使用。

    Spring HttpInvoker的封装

    NULL 博文链接:https://tou3921.iteye.com/blog/1729111

    spring httpInvoker示例

    spring+httpInvoker 实例,含服务端和客户端代码,myEclipse工程,直接导入就可以测试,适合初学者. 提供 普通参数返回值和自定义对象返回值,两个方式.

    spring-httpinvoker-demo

    Spring HTTP Invoker是spring框架中的一个远程调用模型,执行基于HTTP的远程调用,也就是说,可以通过防火墙,并使用java的序列化机制在网络间传递对象。客户端可以很轻松的像调用本地对象一样调用远程服务器上的...

    Spring的HttpInvoker使用示例 (可下载源码)

    NULL 博文链接:https://lggege.iteye.com/blog/369151

    springboot-httpinvoker-demo.zip

    springboot-httpinvoker-demo.zip

    使用httpInvoker提供远程服务

    和Hessian的轻量级传输协议不同的是,Spring HttpInvoker使用Java序列化来序列化参数和返回值,然后基于HTTP协议传输经序列化后的对象。当参数或返回值是复杂类型,并且不能通过Hessian的序列化机制序列化时,...

    基于Spring的HttpInvoker实现改写egova_invoker.zip

    基于Spring的HttpInvoker实现改写服务器端调用: HttpInvoker.invoker 方法,设置InvokerProcess处理客户端调用: ProxyFactory.proxy 方法,生成接口的代理对象,直接调用方法客户端和服务器端的接口和实体类要...

    改造Spring的HttpInvoker, 去掉对Bean的XML配置文件的依赖

    NULL 博文链接:https://lggege.iteye.com/blog/404124

    Java学习之路-Spring的HttpInvoker学习

    Hessian和Burlap都是基于HTTP的,他们...  Spring开发团队意识到RMI服务和基于HTTP的服务之前的空白,Spring的HttpInvoker应运而生。  Spring的HttpInvoker,它基于HTTP之上提供RPC,同时又使用了Java的对象序列化机

    Http Invoker

    接口测试工具,不太好用,不建议使用,自己传着玩儿的

    Java Spring1.2 Remote Invoke HTTP Invoker

    NULL 博文链接:https://lee79.iteye.com/blog/926647

    Java Spring2.5 Remote Invoke HTTP Invoker

    NULL 博文链接:https://lee79.iteye.com/blog/943207

    spring-web-2.5.jar

    org.springframework.remoting.httpinvoker.AbstractHttpInvokerRequestExecutor.class org.springframework.remoting.httpinvoker.CommonsHttpInvokerRequestExecutor.class org.springframework.remoting....

    关于springframework的HttpInvokerProxyFactoryBean和HttpInvokerServiceExporter远程remoting服务项目源码下载.txt

    项目工程结构如下图所示,除了包含spring定义远程服务org.springframework.remoting.httpinvoker.*部分节点代码(红框框部分,具体设计代码部分参考相关代码示例说明文章),还包含springmvc等其他示例部分可以忽略...

    spring jar 包详解

    (9) spring-remoting.jar 这个jar文件包含支持EJB、JMS、远程调用Remoting(RMI、Hessian、Burlap、Http Invoker、JAX-RPC)方面的类。 (10) spring-support.jar 这个jar文件包含支持缓存Cache(ehcache)、JCA、...

Global site tag (gtag.js) - Google Analytics