您的位置:首页 > 编程语言 > Java开发

Spring in Action 学习笔记—第六章远程调用

2006-12-22 16:11 801 查看
[align=center]Spring in Action 学习笔记第六章远程调用[/align]
 
远程调用是客户端应用和服务端之间的会话。在客户端上所需要的一些功能并不包括在该应用的职能范围内。所以应用向能提供这些功能的其他系统寻求帮助。远程的应用通过远程服务把这些功能公开出来。
 
一、Spring远程调用概览
Spring为各种远程访问技术的集成提供了工具类。Spring远程支持是由普通(Spring)POJO实现的,这使得开发具有远程访问功能的服务变得相当容易。
Spring远程调用支持6种不同的RPC模式:远程方法调用(RMI)、Caucho的Hessian和Burlap、Spring自己的HTTP invoker、EJB和使用JAX-RPC 的Web Services。
[align=center]RPC模式[/align]
[align=center]在何种情况下有用[/align]
远程方法调用(RMI)
不考虑网络限制(如防火墙)时,访问/公开基于Java的服务
Hessian或 Burlap
考虑网络限制时,通过HTTP访问/公开基于Java的服务
HTTP invoker
考虑网络限制时,访问/公开基于Spring的服务
EJB
访问用EJB实现的遗留的J2EE系统
JAX-RPC
访问Web Services
其中(来自Spring2.0参考手册):
l         远程方法调用(RMI)。通过使用
RmiProxyFactoryBean和 
RmiServiceExporter,Spring同时支持传统的RMI(使用java.rmi.Remote接口和java.rmi.RemoteException)和通过RMI调用器实现的透明远程调用(支持任何Java接口)。

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

l         Hessian。通过
HessianProxyFactoryBean和 
HessianServiceExporter,可以使用Caucho提供的基于HTTP的轻量级二进制协议来透明地暴露服务。

l         Burlap。 Burlap是Caucho的另外一个子项目,可以作为Hessian基于XML的替代方案。Spring提供了诸如
BurlapProxyFactoryBean和 
BurlapServiceExporter的支持类。

l         JAX RPC。Spring通过JAX-RPC为远程Web服务提供支持。
[align=left]不管选择哪种远程模式,你会发现Spring对每一种模式的支持中贯穿着一个共同的风格。这就意味着你一旦理解了Spring如何配置并使用其中的一种模式,当你决定使用另一种不同的模式的时候,你将拥有非常低的学习曲线。[/align]
在所有的模式中,服务可以作为Spring管理的Bean配置到你的应用中。这是用一个代理工厂Bean实现的,这个Bean使你能把远程服务当作本地对象一样置入到其他Bean的属性中。
客户端发起对代理的调用,好像是代理提供了这些服务的功能一样。代理代表客户端和远程服务交流。它处理连接的具体情况,并向远程服务发起远程调用。
在服务端,你能够把任何Spring管理的Bean的功能公开成为一个远程服务,可使用在表6.1中所列的任何模式(除了EJB和JAX-RPC)。
[align=left]不论开发的是使用远程服务的代码,还是实现那些服务的代码,或者二者兼而有之,在Spring中,使用远程服务纯粹是个配置问题。你不用写任何Java代码来支持远程调用。你的服务Bean不必关心它们是否被卷入到RPC里(虽然任何传递给远程调用的Bean或从远程调用返回的Bean可能需要实现java.io.Serializable)。[/align]
 
二、与RMI一起工作
1.连接RMI服务
Spring的RmiProxyFactoryBean是一个工厂Bean,能创建一个指向RMI服务的代理。用RmiProxyFactoryBean来引用一个RMI PaymentService是非常简单的,只要在Spring配置文件中声明下面的<bean>:
[align=left]<bean id="paymentService" class="org.springframework.remoting.rmi.RmiProxyFactoryBean">[/align]
[align=left] [/align]
[align=left]    <property name="serviceUrl">[/align]
[align=left] [/align]
[align=left]      <value>rmi://${paymenthost}/PayService</value>[/align]
[align=left] [/align]
[align=left]    </property>[/align]
[align=left] [/align]
[align=left]    <property name="serviceInterface">[/align]
[align=left] [/align]
[align=left]      <value>com.springinaction.payment.PaymentService</value>[/align]
[align=left] [/align]
[align=left]    </property>[/align]
[align=left] [/align]
</bean>
[align=left]RMI服务的URL是通过serviceUrl属性设置的。这里,服务被命名为PayService,并且是在一台名字用一个属性占位符配置的机器上(参考第2章的2.4.3节)。serviceInterface属性指明了这个服务所实现的接口,客户端通过这个接口调用在这个服务里的方法。[/align]
把这个支付服务定义为一个Spring管理的Bean,你就能把它作为一个合作者置入到另外的Bean上,就像你对任何非远程的Bean做的那样。举例来说,假设StudentServiceImpl需要使用这个支付服务来批准一张信用卡的支付。你就使用以下代码把RMI服务置入到StudentServiceImpl中:
[align=left]<bean id="studentService"[/align]
[align=left] [/align]
[align=left]       class="com.springinaction.training.service.StudentServiceImpl">[/align]
[align=left] [/align]
[align=left] …[/align]
[align=left] [/align]
[align=left]    <property name="paymentService">[/align]
[align=left] [/align]
[align=left]      <ref bean="paymentService"/>[/align]
[align=left] [/align]
[align=left]    </property>[/align]
[align=left] [/align]
[align=left] …[/align]
[align=left] [/align]
 </bean>
StudentServiceImpl甚至不需要知道它处理的是一个RMI服务。它只是通过注入机制接收PaymentService对象,不必关心它是从哪里来的。此外,代理会捕获任何可能被这个服务抛出的RemoteException,并把它们作为运行时间异常重新抛出,这样,你可以安全地忽略这些异常。这也让远程服务Bean和这个服务的另外实现之间的交换成为可能——或许是不同的远程服务,或者有可能是单元测试时的一个模拟实现。
2.输出RMI服务
Spring提供了比较简单的发布RMI服务的方法:使用POJO。开始之前,你需要写这个服务的接口:
[align=left]publicinterface PaymentService[/align]
[align=left]       {[/align]
[align=left]           public String authorizeCreditCard(String cardNumber, String cardHolderName, int expireMonth, int expireYear, float amount) throws AuthorizationException;[/align]
[align=left] [/align]
[align=left]           publicvoid settlePayment(String authCode, int merchantNumber, float amount) throws SettlementException;[/align]
       }
[align=left]由于服务接口不是从java.rmi.Remote继承的,它的方法都不抛出java.rmi.RemoteException,这点让这个接口简短了很多。但更重要的是,客户端通过这个接口访问支付服务时,将不再需要捕捉那些它们可能没法处理的异常了。下一步,定义服务的实现类:[/align]
[align=left]publicclass PaymentServiceImpl implements PaymentService[/align]
[align=left]       {[/align]
[align=left]           public PaymentServiceImpl() {}[/align]
[align=left] [/align]
[align=left]           public String authorizeCreditCard(String creditCardNumber, String cardHolderName, int expirationMonth, int expirationYear, float amount) throws AuthorizationException[/align]
[align=left]           {[/align]
[align=left]                  // String authCode = ...;[/align]
[align=left]                  // implement authorization[/align]
[align=left]                  return authCode;[/align]
[align=left]           }[/align]
[align=left] [/align]
[align=left]           publicvoid settlePayment(String authCode, int accountNumber, float amount) throws SettlementException[/align]
[align=left]           {[/align]
[align=left]             // implement settlement[/align]
[align=left]           }[/align]
        }
[align=left]你要做的下一件事就是在Spring的配置文件里把PaymentServiceImpl配置为一个<bean>:[/align]
[align=left]<bean id="paymentService" class="org.springframework.payment.PaymentServiceImpl">[/align]
[align=left] …[/align]
</bean>
PaymentServiceImpl没有设置RMI所固有的特性。它仅仅是一个适合在Spring配置文件中声明的简单的POJO。事实上,完全有可能在非远程方式中,通过直接把它置入到客户端里,来使用这个实现。
 
三、使用HessianBurlap的远程调用
[align=left]Hessian和Burlap是Caucho Technology(http://www.caucho.com)提供的两种解决方法,是基于HTTP的轻量级远程服务。它们都致力于通过把它们的API和通信协议变得尽可能的简单,来简化Web服务。[/align]
事实上,Hessian和Burlap是同一个问题的两个方面,但每个都服务于略微不同的目的。Hessian,像RMI那样,使用二进制消息来建立客户端和服务端之间的交流。但与其他二进制远程技术(如RMI)不同的是,它的二进制消息可以移植到其他非Java的语言中。
Burlap是一种基于XML的远程技术,这使得它自然而然地可以移植到任何可以解析XML的语言中。正由于它的XML,比起Hessian的二进制格式来,它的可读性更强。但和其他基于XML的远程技术(例如SOAP或XML-RPC)不同,Burlap的消息结构是尽可能的简单,不需要额外的外部定义语言(如WSDL或IDL等)[1]。
如何在Hessian和Burlap之间做选择。很大程度上说,它们是一样的。惟一的不同就是Hessian的消息是二进制的,而Burlap的消息是XML。由于Hessian的消息是二进制的,所以它在带宽上更占优势。但如果可读性对你来说很重要的话(如出于调试的目的)或者你的应用将和没有Hessian实现(任何除了Java或Python)的语言交流,那么Burlap的XML消息会是更好的选择。
1.访问Hessian/Burlap服务
[align=left]所有RMI的细节都包含在Spring配置文件的Bean的配置里。这样做的好处就是,由于客户端忽略了服务的实现,从一个RMI客户端转到Hessian客户端是极其简单的,不需要改变任何客户端代码。[/align]
[align=left]坏处就是,如果你真地喜欢写代码的话,那么这一节就可能让你有点儿失望了。因为写基于RMI服务的客户端代码和基于Hessian服务的客户端代码惟一的不同就是你将使用Spring的HessianProxyFactoryBean来代替RmiProxyFactoryBean。客户端代码中基于Hessian的支付服务可以用以下代码声明:[/align]
[align=left]<bean id="paymentService" class="org.springframework.[/align]
[align=left] [/align]
[align=left]          ➥remoting.caucho.HessianProxyFactoryBean">[/align]
[align=left] [/align]
[align=left]    <property name="serviceUrl">[/align]
[align=left] [/align]
[align=left]        <value>http://${serverName}/${contextPath}/pay.service</value>[/align]
[align=left] [/align]
[align=left]    </property>[/align]
[align=left] [/align]
[align=left]    <property name="serviceInterface">[/align]
[align=left] [/align]
[align=left]        <value>com.springinaction.payment.PaymentService</value>[/align]
[align=left] [/align]
[align=left]    </property>[/align]
[align=left] [/align]
 </bean>
[align=left]就像基于RMI的服务那样,serviceInterface属性指定这个服务实现的接口。并且,如同RmiProxyFactoryBean,serviceUrl表明这个服务的URL。既然Hessian是基于HTTP的,当然应该在这里设置一个HTTP URL了(你将在下一节中看到这个URL是如何得来的)。[/align]
[align=left]事实证明,写一个Burlap服务是同样无趣的。二者惟一的不同就是,你使用BurlapProxyFactoryBean代替HessianProxyFactoryBean:[/align]
[align=left]<bean id="paymentService" class="org.springframework.[/align]
[align=left] [/align]
[align=left]          ➥remoting.caucho.BurlapProxyFactoryBean">[/align]
[align=left] [/align]
[align=left]    <property name="serviceUrl">[/align]
[align=left] [/align]
[align=left]      <value>http://${serverName}/${contextPath}/pay.service</value>[/align]
[align=left] [/align]
[align=left]    </property>[/align]
[align=left] [/align]
[align=left]    <property name="serviceInterface">[/align]
[align=left] [/align]
[align=left]      <value>com.springinaction.payment.PaymentService</value>[/align]
[align=left] [/align]
[align=left]    </property>[/align]
[align=left] [/align]
 </bean>
尽管我们觉得在RMI、Hessian和Burlap服务之间稍微不同的配置是很没有乐趣的,但这个单调恰恰是有好处的。它意味着你不费吹灰之力就可以在各种Spring支持的远程技术之间转换,无须去学习一个全新的模型。你一旦配置了一个对RMI服务的引用,把它重新配置为Hessian或Burlap服务也是很轻松的工作。
2.用HessianBurlap公开Bean的功能
输出一个Hessian服务:
在Spring里输出一个Hessian服务和在Spring里实现一个RMI服务惊人地相似。为把支付服务公开为RMI服务,你得在Spring配置文件中配置一个RmiServiceExporter Bean。非常类似的,把支付服务公开为Hessian服务,你也需要配置一个exporter Bean。只不过这一次用的是HessianServiceExporter:
[align=left]<bean name="hessianPaymentService" class="org.springframework.[/align]
[align=left] [/align]
[align=left]         ➥remoting.caucho.HessianServiceExporter">[/align]
[align=left] [/align]
[align=left]    <property name="service">[/align]
[align=left] [/align]
[align=left]      <ref bean="paymentService"/>[/align]
[align=left] [/align]
[align=left]    </property>[/align]
[align=left] [/align]
[align=left]    <property name="serviceInterface">[/align]
[align=left] [/align]
[align=left]      <value>com.springinaction.payment.PaymentService</value>[/align]
[align=left] [/align]
[align=left]    </property>[/align]
[align=left] [/align]
 </bean>
[align=left]HessianServiceExporter在Hessian服务中实现的功能和RmiServiceExporter在RMI服务中的功能是完全一样的。那就是说,它把一个Bean的公共方法公开为一个Hessian服务的方法。[/align]
[align=left]正如RmiServiceExporter,service属性中被置入了实现这个服务的Bean的引用。这里,这个service属性绑定的是paymentService Bean的引用。serviceInterface属性用来表示PaymentService是这个服务所实现的接口。[/align]
[align=left]然而,和RmiServiceExporter不同,你不需要设置serviceName属性。在RMI中,serviceName属性用来在RMI注册表中注册一个服务。Hessian没有注册表,因此就没有必要命名一个Hessian服务。[/align]
[align=left]配置Hessian控制器:[/align]
[align=left]RmiServiceExporter和HessianServiceExporter另外一个主要的区别就是,由于Hessian是基于HTTP的,所以HessianServiceExporter被实现成Spring MVC的Controller。这就是说,为了使用输出的Hessian服务,你需要完成两个额外的配置步骤:[/align]
[align=left]1.在你的Spring配置文件中配置一个URL处理器,来分发Hessian服务的URL给适当的Hessian服务Bean。[/align]
[align=left]2.在web.xml中配置一个Spring的DispatcherServlet,并把你的应用部署为web应用。[/align]
输出一个Burlap服务:
把Spring管理的Bean作为Burlap服务输出。用Spring的BurlapServiceExporter来代替HessianServiceExporter就能完成这项任务:
[align=left]<bean name="burlapPaymentService" class="org.springframework.[/align]
[align=left] [/align]
[align=left]            ➥remoting.caucho.BurlapServiceExporter">[/align]
[align=left] [/align]
[align=left]    <property name="service">[/align]
[align=left] [/align]
[align=left]      <ref bean="paymentService"/>[/align]
[align=left] [/align]
[align=left]    </property>[/align]
[align=left] [/align]
[align=left]    <property name="serviceInterface">[/align]
[align=left] [/align]
[align=left]      <value>com.springinaction.payment.PaymentService</value>[/align]
[align=left] [/align]
[align=left]    </property>[/align]
[align=left] [/align]
 </bean>
[align=left]你会发现,除了Bean的名字(完全是任意的)和使用了BurlapServiceExporter以外,这个Bean和hessianPaymentService是一样的。配置Burlap服务和配置Hessian服务的其他方面也是一样的,这也就包括了需要建一个URL处理器和DispatcherServlet。Hessian和Burlap解决了RMI头疼的防火墙问题。[/align]
 
四、使用HTTP invoker
1.通过HTTP访问服务
[align=left]访问一个HTTP invoker服务,你需要使用HttpInvokerProxyFactoryBean。要让支付服务作为一个HTTP invoker服务公开,得配置一个Bean,用HttpInvokerProxyFactoryBean来代理它,如下所示:[/align]
[align=left]<bean id="paymentService" class= "org.springframework.remoting.[/align]
[align=left] [/align]
[align=left]          ➥httpinvoker.HttpInvokerProxyFactoryBean">[/align]
[align=left] [/align]
[align=left]    <property name="serviceUrl">[/align]
[align=left] [/align]
[align=left]      <value>http://${serverName}/${contextPath}/pay.service</value>[/align]
[align=left] [/align]
[align=left]    </property>[/align]
[align=left] [/align]
[align=left]    <property name="serviceInterface">[/align]
[align=left] [/align]
[align=left]      <value>com.springinaction.payment.PaymentService</value>[/align]
[align=left] [/align]
[align=left]    </property>[/align]
[align=left] [/align]
 </bean>
serviceInterface属性仍然用来表示这个支付服务所实现的接口;serviceUrl属性仍然是用来表示远程支付服务的位置。由于HTTP invoker是基于HTTP的,如同Hessian和Burlap一样,serviceUrl就能包含与Hessian和Burlap版本的Bean里一样的URL。
2.把Bean作为HTTP服务公开
使用HttpInvokerServiceExporter把Bean的方法输出为远程方法,面的Bean的定义展示了如何把paymentService Bean作为一个远程的基于HTTP invoker的服务输出:
[align=left]<bean id="httpPaymentService" class="org.springframework.remoting.[/align]
[align=left] [/align]
[align=left]        ➥httpinvoker.HttpInvokerServiceExporter">[/align]
[align=left] [/align]
[align=left]    <property name="service">[/align]
[align=left] [/align]
[align=left]      <ref bean="paymentService"/>[/align]
[align=left] [/align]
[align=left]    </property>[/align]
[align=left] [/align]
[align=left]    <property name="serviceInterface">[/align]
[align=left] [/align]
[align=left]      <value>com.springinaction.payment.PaymentService</value>[/align]
[align=left] [/align]
[align=left]    </property>[/align]
[align=left] [/align]
</bean>
[align=left]基于HTTP invoker的服务,顾名思义,是基于HTTP的,就像Hessian和Burlap服务一样。并且,也和HessianServiceExporter和BurlapServiceExporter那样,HttpInvokerServiceExporter也是一个Spring的Controller。这就意味着你需要建立一个URL处理器,把HTTP URL映射到服务上:[/align]
[align=left]<bean id="urlMapping" class="org.springframework.web.servlet.handler.SimpleUrlHandlerMapping">[/align]
[align=left] [/align]
[align=left]    <property name="mappings">[/align]
[align=left] [/align]
[align=left]      <props>[/align]
[align=left] [/align]
[align=left]         <prop key="/pay.service">httpPaymentService</prop>[/align]
[align=left] [/align]
[align=left]      </props>[/align]
[align=left] [/align]
[align=left]    </property>[/align]
[align=left] [/align]
 </bean>
[align=left]Spring的HTTP invoker是作为一个两全其美的远程调用解决方案出现的,把HTTP交流的简单性和Java内置的对象序列化机制结合起来。这让HTTP invoker服务成为一个引人注目的对RMI或是对Hessian/Burlap的替代品。[/align]
[align=left]要记住HTTP invoker有个重大的限制,它是一个只在Spring框架中提供的远程调用解决方案。这就意味着客户端和服务器端都必须是使用Spring的应用。[/align]
 
五.使用EJB
[align=left]虽然Spring提供了大量的功能,让POJO具有EJB的能力,但你或许不能总是享受在完全没有EJB的项目上工作的奢侈。一方面,你可能会接触一些其他系统,它们的功能是通过无状态的会话EJB开放出来的。另一方面,你可能被放在一个项目中,由于正统技术(或者可能是政治)的原因,你不得不写EJB代码。[/align]
[align=left]不管你的应用是EJB客户端,还是你必须写EJB本身,你都不需要为了用EJB,完全放弃Spring带来的好处。Spring有两种方法提供对EJB的支持:[/align]
[align=left]   Spring能让你在Spring的配置文件里,把EJB作为Bean来声明。这样,把EJB引用置入到其他Bean的属性里就成为可能了,好像EJB就是另一个POJO。[/align]
[align=left]   Spring能让你写EJB,让EJB成为Spring配置的Bean的代理的工作。[/align]
 
六、使用JAX-RPCWeb Service
JAX-RPC是“基于XML的远程调用的Java API(Java APIs for XML-based remote procedure call)”的缩写。这是一个口语化的词,仅仅意味着JAX-RPC是Java程序使用XML访问远程服务的一种方式。特别地,这个服务是指用SOAP(Simple Object Access Protocol)协议公开它们的功能的web service。
 
七、小结
[align=left]Spring提供了远程服务的支持,让使用远程服务和使用常规的JavaBean一样简单。[/align]
[align=left]在客户端,Spring提供了代理工厂Bean,能让你在Spring应用中配置远程服务。不管是使用RMI、Hessian、Burlap、HTTP invoker、EJB、还是Web service,你都可以把远程服务置入到你的应用里,好像它们是POJO一样。Spring甚至捕获了所有抛出的RemoteException,并在发生异常的地方重新抛出运行时RemoteAccessException,让你的代码从处理可能不可恢复的异常中解放出来。[/align]
    Spring的远程调用我不知什么时候才能用上,在这里在不太理解的基础上对书作了简单的摘抄,先留个记号等以后用着了在回头学习。
 
 
 

Trackback: http://tb.blog.csdn.net/TrackBack.aspx?PostId=1423884
 
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
相关文章推荐