RMI

RMI远程方法调用允许在一台机器上执行的Java对象调用另一台机器上运行的Java对象,它允许我们构建分布式的应用程序。它的方法被包含在java.rmi包中

构建RMI程序的通常形式:

1。定义远程接口(扩展Remote接口,Remote没有成员,只是指明一个接口使用远程方法),主要是对要实现的功能在接口里说明,方法都抛出RemoteException异常。

2。实现远程接口(扩展/继承UnicastRemoteObject类,实现上面的远程接口)

3。服务器端程序:更新机器的RMI注册表。通过Naming的rebind方法,把名字和对象引用关联到一起。

4。客户端程序:这里的服务器URL必须使用RMI协议,字符串包括IP或名字以及字符串“AddServer”,然后调用Naming类的lookup方法。

实现远程加法计算程序例子:

//远程接口
import java.rmi.*;
public interface AddServerInterface extends Remote{
 double add(double d1, double d2) throws RemoteException;
}

//远程实现
import java.rmi.*;
import java.rmi.server.*;
public class AddServerImplement  extends UnicastRemoteObject
 implements AddServerInterface{
 public AddServerImplement() throws RemoteException{
 } 
 public double add(double d1, double d2) throws RemoteException{
  return d1 + d2;
 }
}

//服务器
import java.net.*;
import java.rmi.*;
public class AddServer {
 public static void main(String[] args) {
  try{
   AddServerImplement addsi = new AddServerImplement();
   Naming.rebind("AddServer", addsi);
  }catch(Exception e){
   System.out.println("Exception : " + e);
  }
 } 
}

//客户端
import java.rmi.*;
public class AddClient {
 public static void main(String[] args) {
  try{
   String addServerURL = "rmi://" + args[0] + "/AddServer";
   AddServerInterface addsi = (AddServerInterface)Naming.lookup(addServerURL);
   
   System.out.println("The first number is:" + args[1]);
   System.out.println("The sencond number is:" + args[2]);
   double d1 = Double.valueOf(args[1]).doubleValue();
   double d2 = Double.valueOf(args[2]).doubleValue();
   
   System.out.println("The sum is:" + addsi.add(d1,d2));   
  }catch(Exception e){
   e.printStackTrace();
  }
  
 } 
}

程序完成后,需要生成存根(Stub,驻留于客户机上的Java对象,提供与远程服务同样的接口)和基干(Skeleton,Java2中不是必须,它是驻留于服务器上的Java对象)。用rmic AddServerImplement命令生成。(远程方法可接受简单类型的数据和对象作为参数。作为参数传递给远程方法的对象必须序列化(根据序列化的理论,序列化工具也会处理该对象的所有的引用对象))

生成Stub和Skeleton之后,需要把客户端(AddClient)、远程接口(AddServerInterface)、远程接口的基干(AddServerImplement_Stub)放在同一目录。

然后start rmiregistry 启动RMI注册表

接下来就是启动服务器和客户机了



 

 

03-24
### Java RMI 教程及相关问题 远程方法调用 (Remote Method Invocation, RMI) 是一种用于实现客户端和服务器之间通信的技术。它允许对象在不同的虚拟机上运行并相互交互,就像它们在同一进程中一样[^1]。 #### RMI 的基本概念 RMI 提供了一种机制,使得程序可以像调用本地方法那样调用远程对象上的方法。这种技术依赖于序列化来传输数据,并通过动态代理类支持接口的透明调用[^3]。 以下是 RMI 工作的核心组件: - **Stub 和 Skeleton**: Stub 负责接收来自客户端的方法调用并将请求转发给远程对象;Skeleton 则负责将这些请求传递到实际的对象实例。 - **Registry**: 远程对象注册表是一个服务,用来存储远程对象的引用以便客户端能够查找并连接到它们。 #### 创建简单的 RMI 应用程序 要创建一个基于 RMI 的应用程序,通常需要以下几个步骤: 1. 定义远程接口:该接口必须扩展 `java.rmi.Remote` 接口,并声明所有可被远程访问的方法为抛出 `java.rmi.RemoteException` 的抽象方法。 ```java import java.rmi.*; public interface Hello extends Remote { String sayHello() throws RemoteException; } ``` 2. 实现远程接口:编写具体的实现类继承自 `UnicastRemoteObject` 并提供接口定义的功能。 ```java import java.rmi.server.UnicastRemoteObject; import java.rmi.RemoteException; public class HelloImpl extends UnicastRemoteObject implements Hello { protected HelloImpl() throws RemoteException { super(); } @Override public String sayHello() throws RemoteException { return "Hello, world!"; } } ``` 3. 启动命名服务(rmiregistry),绑定远程对象至注册中心。 ```bash rmiregistry & ``` 4. 编写服务器端代码以启动服务并与注册表关联。 ```java import java.rmi.registry.LocateRegistry; import java.rmi.registry.Registry; public class Server { public static void main(String args[]) { try { Registry registry = LocateRegistry.createRegistry(1099); Hello helloObj = new HelloImpl(); registry.rebind("HelloServer", helloObj); System.out.println("Server ready"); } catch (Exception e) { System.err.println("Server exception: " + e.toString()); e.printStackTrace(); } } } ``` 5. 开发客户端应用去定位远程对象并通过其 stub 来执行操作。 ```java import java.rmi.registry.LocateRegistry; import java.rmi.registry.Registry; public class Client { private Client() {} public static void main(String[] args) { try { Registry registry = LocateRegistry.getRegistry("localhost"); Hello stub = (Hello) registry.lookup("HelloServer"); String response = stub.sayHello(); System.out.println("response: " + response); } catch (Exception e) { System.err.println("Client exception: " + e.toString()); e.printStackTrace(); } } } ``` #### 常见问题排查 如果遇到与 RMI 相关的问题,可以从以下方面入手分析: - 网络配置错误可能导致无法建立连接; - 防火墙阻止了必要的端口通讯; - 类加载器未能正确处理跨 JVM 边界的类版本差异等问题。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值