Java RMI实现以及Spring封装RMI实现小结

来源:互联网 发布:mac os 启动管理界面 编辑:程序博客网 时间:2024/05/16 13:59
  最近应用到了Java RMI技术,所以总结一下RMI的使用,另外EJB的实现也是以Java RMI为基础核心的。RMI远程方法调用,可以跨机器通过网络调用,不过Java RMI只支持两边都是Java程序,如果要实现完全两边异构,那么就要用到传说中的Web Service了。为了看好效果,都要建立两个或两个以上工程,当然,如果你有两个机器并通过局域网相连更好,如果能同有互联网的条件就更好了,以下是同一机器不同工程的实验。

一、Java RMI实现

首先新建一个工程,随便什么工程,为了方便,就JavaProject吧。

1、创建一个接口,继承Remote

[html] view plain copy
  1. package rmi.java;  
  2.   
  3. import java.rmi.Remote;  
  4. import java.rmi.RemoteException;  
  5. /**   
  6.  * 定义远程接口,必须继承Remote接口,   
  7.  * 其中所有需要远程调用的方法都必须抛出RemoteException异常    
  8.  */   
  9. public interface IHello extends Remote {  
  10.       
  11.     public String sayHello(String name) throws RemoteException;     
  12.     public int sum(int a, int b)throws RemoteException;   
  13.   
  14.   
  15. }  

2、创建接口的实现类

[html] view plain copy
  1. package rmi.java;  
  2.   
  3. import java.rmi.RemoteException;  
  4. import java.rmi.server.UnicastRemoteObject;  
  5.   
  6. public class HelloImpl extends UnicastRemoteObject implements IHello{  
  7.       
  8.     private static final long serialVersionUID = 1961558474342609777L;  
  9.       
  10.         public HelloImpl()throws RemoteException {  
  11.         super();  
  12.     }  
  13.   
  14.   
  15.     @Override  
  16.     public String sayHello(String name) {  
  17.         return "Hello, " + name;    
  18.     }  
  19.   
  20.   
  21.     @Override  
  22.     public int sum(int a, int b) {  
  23.           
  24.         return a+b;  
  25.     }  
  26.   
  27. }  

 

       说明:接口的实现类同时要实现Serializable接口,这里继承UnicastRemoteObject也是间接实现Serializable接口,同时,因为构造方法需要抛出RemoteException,所以不能缺省使用隐含的无参构造方法,而应该自己显式定义构造方法。

 

3、创建应用类,注册和启动服务端RMI,以被客户端调用

[html] view plain copy
  1. package rmi.java;  
  2.   
  3. import java.net.MalformedURLException;  
  4. import java.rmi.AlreadyBoundException;  
  5. import java.rmi.Naming;  
  6. import java.rmi.RemoteException;  
  7. import java.rmi.registry.LocateRegistry;  
  8.   
  9.   
  10. public class HelloServer {  
  11.     public static void main(String args[]) {  
  12.         try {  
  13.             //创建一个远程对象  
  14.             IHello rhello = new HelloImpl();      
  15.             //生成远程对象注册表Registry的实例,并指定端口为8888(默认端口是1099)  
  16.             LocateRegistry.createRegistry(8888);  
  17.   
  18.             //把远程对象注册到RMI注册服务器上,并命名为RHello  
  19.             //绑定的URL标准格式为:rmi://host:port/name(协议名可以省略,下面两种写法都可以)  
  20.             Naming.bind("rmi://127.0.0.1:8888/RHello", rhello);  
  21.           
  22.             System.out.println(">>INFO:远程IHello对象绑定成功!");  
  23.         } catch (RemoteException e) {  
  24.             System.out.println("创建远程对象发生异常!");  
  25.             e.printStackTrace();  
  26.         } catch (AlreadyBoundException e) {  
  27.             System.out.println("发生重复绑定对象异常!");  
  28.             e.printStackTrace();  
  29.         } catch (MalformedURLException e) {  
  30.             System.out.println("发生URL畸形异常!");  
  31.             e.printStackTrace();  
  32.         }  
  33.     }  
  34. }  


      说明:绑定的地址10.225.112.86是我的局域网地址可以在DOS命令行用ipconfig查看,如果你的机器没有任何联网,可以使用127.0.0.1或localhost。

      运行HelloServer.Java看到,红色方块显示正在运行:

       >>INFO:远程IHello对象绑定成功!

好了,现在远程服务提供端建立完成,下面建立客户端。

新建一个新的工程,为了方便,也是javaProject吧,

1、 因为客户端需要有服务端那边提供的接口,才可以访问,所以要将服务端的IHello接口完全拷贝(连同包)到客户端,当然为了方便,你在客户端工程中新建一个完全一样的接口也可以。实际运用中通常是要服务端接口打成jar包来提供的。

2、 创建客户端调用类

[html] view plain copy
  1. package rmi.java;  
  2.   
  3. import java.rmi.Naming;  
  4.   
  5.   
  6. public class HelloClient {  
  7.     public static void main(String args[]) {  
  8.         try {  
  9.             // 在RMI服务注册表中查找名称为RHello的对象,并调用其上的方法  
  10.             IHello rhello = (IHello) Naming.lookup("rmi://127.0.0.1:8888/RHello");  
  11.             System.out.println(rhello.sayHello("world"));  
  12.             System.out.println(rhello.sum(454, 5457));  
  13.         } catch (Exception e) {  
  14.             e.printStackTrace();  
  15.         }  
  16.     }  
  17. }  

       运行,成功。

       下面我们要使用spring封装的Java RMI技术,也是很多项目都会用到的。后面我有个Spring RMI的例子。要看懂下面Spring的例子,你需要已经会用Spring,会配置Spring,否则怕你看不懂,所以如果还不懂Spring的,先学学Spring,入下门吧。

二、Spring 封装RMI实现

       Spring RMI中,主要有两个类:org.springframework.remoting.rmi.RmiServiceExporterorg.springframework.remoting.rmi.RmiProxyFactoryBean

服务端使用RmiServiceExporter暴露RMI远程方法,客户端用RmiProxyFactoryBean间接调用远程方法。

 

       首先,也是两个工程,服务端用Web工程,因为使用Spring,我们依托Web容器来完成。

       1、在该服务端Web工程中添加接口,普通接口,无需继承其他

[html] view plain copy
  1. package rmi.test;  
  2.   
  3. public interface HelloRMIService {  
  4.       
  5.     public int  getAdd(int a, int b);  
  6.        
  7.   
  8. }  


 

        2、接口的实现类

[html] view plain copy
  1. package rmi.test;  
  2.   
  3. public class HelloRMIServiceImpl implements HelloRMIService {  
  4.   
  5.     @Override  
  6.     public int getAdd(int a, int b) {  
  7.           
  8.         return a+b;  
  9.     }  
  10.   
  11. }  


 

       3、在该服务端Web工程中添加Spring的bean配置文件,比如命名为rmiServer.xml,内容如下:
[html] view plain copy
  1. <?xml version="1.0" encoding="UTF-8" standalone="no"?>  
  2. <beans xmlns="http://www.springframework.org/schema/beans"  
  3.        xmlns:aop="http://www.springframework.org/schema/aop"  
  4.        xmlns:context="http://www.springframework.org/schema/context"  
  5.        xmlns:jee="http://www.springframework.org/schema/jee"   
  6.        xmlns:tx="http://www.springframework.org/schema/tx"  
  7.        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"  
  8.        xsi:schemaLocation="http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-3.0.xsd     
  9.        http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.0.xsd     
  10.        http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.0.xsd     
  11.        http://www.springframework.org/schema/jee http://www.springframework.org/schema/jee/spring-jee-3.0.xsd     
  12.        http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-3.0.xsd">  
  13.   
  14.     <bean id="helloRMIServiceImpl" class="rmi.test.HelloRMIServiceImpl"> </bean>  
  15.     <!-- 将一个类发布为一个RMI服务 -->  
  16.     <bean id="myRMIServer"  class="org.springframework.remoting.rmi.RmiServiceExporter">  
  17.       <property name="serviceName" value="helloRMI"></property>  
  18.       <property name="service" ref="helloRMIServiceImpl"></property>  
  19.       <property name="serviceInterface" value="rmi.test.HelloRMIService"></property>  
  20.       <property name="registryPort" value="9999"></property>  
  21.     </bean>  
  22.       
  23.     
  24.       
  25. </beans>  

       说明:这里不详细的说明了,主要配置了真实实现类,用RmiServiceExporter暴露时,配置property要注意的有service,serviceName,serviceInterface,端口registryPort。

       启动Web工程的服务器,该配置文件应该被Spring的监听器监听,并加载,启动成功后,服务端就算建好了。如果服务器是在localhost启动的,那么暴露的RMIIP也是localhost,如果需要使用其他IP,需要让服务器在其他的IP启动。

    这里没有web服务器,使用main模拟启动,如下:

[html] view plain copy
  1. package rmi.test;  
  2.   
  3. import org.springframework.context.support.ClassPathXmlApplicationContext;  
  4.   
  5. public class RMIServiceTest {  
  6.       
  7.     public static void main(String[] args) {  
  8.           
  9.         new ClassPathXmlApplicationContext("rmiServer.xml");  
  10.           
  11.     }  
  12.   
  13. }  


     客户端调用:为了方便也只新建一个简单的Java Project,使用静态的java代码来调用了。

     1、 在源文件src下建立一个rmiClient.xml

[html] view plain copy
  1. <?xml version="1.0" encoding="UTF-8" standalone="no"?>  
  2. <beans xmlns="http://www.springframework.org/schema/beans"  
  3.        xmlns:aop="http://www.springframework.org/schema/aop"  
  4.        xmlns:context="http://www.springframework.org/schema/context"  
  5.        xmlns:jee="http://www.springframework.org/schema/jee"   
  6.        xmlns:tx="http://www.springframework.org/schema/tx"  
  7.        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"  
  8.        xsi:schemaLocation="http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-3.0.xsd     
  9.        http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.0.xsd     
  10.        http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.0.xsd     
  11.        http://www.springframework.org/schema/jee http://www.springframework.org/schema/jee/spring-jee-3.0.xsd     
  12.        http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-3.0.xsd">  
  13.   
  14.     <!-- 客户端 -->  
  15.     <bean id="myRMIClient"  class="org.springframework.remoting.rmi.RmiProxyFactoryBean">  
  16.       <property name="serviceInterface" value="rmi.test.HelloRMIService"></property>  
  17.       <property name="serviceUrl" value="rmi://127.0.0.1:9999/helloRMI"></property>  
  18.     </bean>  
  19.       
  20.          
  21. </beans>  

 

       这里注意到RmiProxyFactoryBean的两个重要的property:serviceUrlserviceInterface,HelloRMIService接口可以从服务端的接口打成jar包来提供。

        2、 新建java类

[html] view plain copy
  1. package rmi.test;  
  2.   
  3. import org.springframework.context.ApplicationContext;  
  4. import org.springframework.context.support.ClassPathXmlApplicationContext;  
  5.   
  6. public class RMIClient {  
  7.       
  8.     public static void main(String[] args) {  
  9.         ApplicationContext applicationContext = new ClassPathXmlApplicationContext("rmi/test/rmiClient.xml");  
  10.         HelloRMIService helloRMIService =  applicationContext.getBean("myRMIClient",HelloRMIService.class);  
  11.         System.out.println(helloRMIService.getAdd(3, 4));  
  12.     }  
  13.   
  14. }  

        运行,成功!好了,这就是一个采用Spring封装的RMI的例子,项目工作中应该经常使用的。有什么问题,欢迎讨论交流。
原创粉丝点击