`

Spring HTTP invoker RPC

 
阅读更多

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.       访问服务方法

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

在jsp,servlet,action等等文件中

UcService service = (UcService) WebApplicationContextUtils

        .getRequiredWebApplicationContext(

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

            "httpService");

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

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

ApplicationContext applicationContextnew FileSystemXmlApplicationContext("classpath:application-context.xml");

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

依赖注入,远程调用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这个处理适配器

另外,默认情况下,客户端的HttpInvokerProxy使用J2SE的HTTP Client来建立连接,即org.springframework.remoting.httpinvoker.SimpleHttpInvokerRequestExecutor,可以通过设置httpInvokerRequestExecutor属性来改变默认配置,spring提供了另外一种HttpClient,org.springframework.remoting.httpinvoker.CommonsHttpInvokerRequestExecutor。

修改配置如下:

<bean id="httpService"
  class="org.springframework.remoting.httpinvoker.HttpInvokerProxyFactoryBean">
  <property name="serviceUrl">
   <value>http://vm.netqin.com:4080/ucs/service/httpService</value>
  </property>
  <property name="serviceInterface" value="com.netqin.baike.service.UcService">
  </property>
  <property name="httpInvokerRequestExecutor">
   <bean
    class="org.springframework.remoting.httpinvoker.CommonsHttpInvokerRequestExecutor" />
  </property>
 </bean>

需要在项目中引入两个jar包:

commons-codec-x.x.jar

commons-httpclient-x.x.x.jar

CommonsHttpInvokerRequestExecutor具有HTTP connection pooling,不过通过使用jmeter进行压力测试发现,SimpleHttpInvokerRequestExecutor性能高于CommonsHttpInvokerRequestExecutor

转至:http://blog.csdn.net/hanqunfeng/article/details/4303127

分享到:
评论

相关推荐

    spring-httpinvoker-demo

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

    springboot-httpinvoker-demo.zip

    《SpringBoot HTTP ...尽管HTTP Invoker在性能上可能不如专门的RPC框架,但它简单易用,适合于小型项目或对性能要求不高的场景。在理解了其工作原理和配置方式后,开发者可以根据自身需求灵活运用,提高开发效率。

    http invoker 做post测试

    HTTP Invoker 是一个Java框架,主要用于在分布式系统中进行远程方法调用(RPC)。它通过HTTP协议提供服务,使得客户端可以像调用本地方法一样调用远程服务。在这个场景下,“http invoker 做post测试”指的是在完成...

    Java学习之路-Spring的HttpInvoker学习

    Hessian和Burlap都是基于HTTP的,他们都解决了RMI所头疼的防火墙渗透问题。但当传递过来的RPC消息中包含序列化对象时,RMI完胜...  Spring的HttpInvoker,它基于HTTP之上提供RPC,同时又使用了Java的对象序列化机

    Java Spring1.2 Remote Invoke HTTP Invoker

    Java Spring 1.2 远程...尽管在某些场景下,现代的RPC框架如gRPC或RESTful API可能更受欢迎,但HTTP Invoker仍然在许多现有项目中扮演着重要角色。理解并熟练掌握其工作原理和配置,对于开发和维护这些系统至关重要。

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

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

    用Http Invoker实现RCP客户端与后台的交互

    `Http Invoker`是Spring框架提供的一种基于HTTP协议的RPC实现,主要用于Java应用程序之间进行服务调用。本文将深入探讨如何使用Http Invoker实现RCP客户端与后台的交互,以及相关知识点。 1. **Http Invoker的基本...

    Hessian 、 HttpInvoker 、 XFire 、 Axis

    HttpInvoker主要面向Spring应用程序,易于集成,但相比Hessian,其数据传输效率较低,因为HTTP协议本身较重,且Java序列化也相对消耗资源。 XFire(后来被Apache CXF吸收)是一款基于XML的Web服务框架,它支持多种...

    Eclipse工程Http Invoker Service & Client

    Http Invoker是Spring框架的一个组件,允许在分布式环境中通过HTTP协议透明地调用Java对象的方法,提供了一种轻量级的RPC(Remote Procedure Call)实现。 **Http Invoker服务端(Service)** 在Http Invoker...

    使用spring远程调用服务端接口实现WebService功能

    &lt;bean id="httpInvokerExporter" class="org.springframework.remoting.httpinvoker.HttpInvokerServiceExporter"&gt; ``` 这将把`MyRemoteService`接口暴露为HTTP服务,客户端可以通过HTTP请求调用`doSomething`...

    spring远程调用

    HttpInvoker是Spring框架的一部分,它基于HTTP协议实现了远程过程调用(RPC)。与RMI(Remote Method Invocation)相比,HttpInvoker更适合于跨越防火墙的场景,因为它使用的是标准的HTTP协议,而RMI可能被防火墙...

    基于Spring的RPC通讯模型的使用与比较

    HTTP Invoker是Spring自带的RPC解决方案,它使用HTTP协议来传输RPC调用。客户端可以使用HttpInvokerProxyFactoryBean来创建代理,并使用该代理来调用远程服务。 三、客户端和服务端的实现 客户端: * 客户端需要...

    http接口相关介绍

    与传统的远程过程调用(RPC)不同,Spring HTTP Invoker简化了客户端和服务端之间的交互,使得开发者能够像调用本地对象一样调用远程服务。 ##### 2.1 服务模式说明 - **服务器端**:通过HTTP Invoker服务暴露服务...

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

    2. Spring的HTTP调用器(Spring’s Http Invoker): Spring提供了一种特殊的允许通过HTTP进行Java串行化的远程调用策略,支持任意Java接口(就像RMI调用器)。相对应的支持类是 HttpInvokerProxyFactoryBean和 ...

    基于Spring的远程访问与Web Service

    标题与描述中提到的主题是“基于Spring的远程访问与Web Service”,这涉及到Spring框架在企业级应用中的关键特性——即其支持多种远程服务调用的方式,包括但不限于RMI、HTTP Invoker、Hessian、Burlap、JAX-RPC以及...

    spring jar 包详解

    - **功能简介**:包含了支持 EJB、JMS、远程调用 Remoting(RMI、Hessian、Burlap、Http Invoker、JAX-RPC)方面的类。 - **应用场景**:适用于需要实现远程调用功能的项目。 - **依赖关系**:依赖于 `spring-core....

    Spring 2.5 jar 所有开发包及完整文档及项目开发实例

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

    最新最全的spring开发包

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

    Spring包结构以及各个包之间引用关系说明.doc

    - 提供了对 EJB、JMS 以及远程调用(如 RMI、Hessian、Burlap、HttpInvoker、JAX-RPC)相关的类。 10. **spring-support.jar** - 该组件提供了对 Cache(如 ehcache)、JCA、JMX、邮件服务(如 JavaMail、COS ...

    Spring应用开发实战Web Service WS

    Spring框架支持多种远程访问技术,包括RMI(Remote Method Invocation)、JAX-RPC(Java API for XML-based Remote Procedure Call)、Hessian、Burlap、XFire以及Spring自带的HTTPInvoker。这些技术都有各自的使用...

Global site tag (gtag.js) - Google Analytics