Remote Method Invocation(RMI)即远程方法调用,是Java远程访问的规范之一。
RMI本质就是代理模式的应用。
RMI可以为我们处理多线程、网络通信等细节,因此使用RMI开发网络应用格外简单,没有多线程处理,没有繁琐的IO处理,只需定义合适的远程方法调用即可。
CORBA(Common Object Request Broker Architecture,通用对象请求代理架构)、Web Service等,可实现类似功能。
不足:
1:同步通信。
2:客户端和服务器端生命周期耦合。
3:点对点通信。
解决方法:使用面向消息的应用架构,其可实现“完全解耦”,如JMS,MDB。
开发RMI服务器
步骤:
1:开发RMI服务器接口必须实现java.rmi.Remote接口,所有在java.rmi.Remote接口里的方法必须抛出RemoteException异常。
2:为服务器接口编写实现类,该类必须继承java.rmi.server.UnicastRemoteObject。
3:编写实现类的main()方法。main()需要创建服务器类的实例,并通过Naming类的bind()或rebind()方法将其绑定到指定的JNDI名称。
4:在服务器上注册RMI服务:
方法一:JDK为此提供了rmiregistry工具,运行该工具的命令:
rmiregistry <port>
RMI默认服务端口是1099。
方法二:在服务器实现类的main()方法中注册,使用
LocateRegistry.createRegistry(1099);
发开RMI客户端
步骤:
1:通过JNDI查找提供远程RMI服务的对象,并对其执行强制类型转换。
2:调用远程方法。
3:将Server接口的class文件复制到客户端中。
示例:
Server.java
import java.rmi.*;
public interface Server extends Remote {
String helloWorld(String name)
throws RemoteException;
Person getPerson(String name , int age)
throws RemoteException;
}
Person.java
import java.io.*;
public class Person
implements Serializable {
private String name;
private int age;
public Person() {}
public Person(String name , int age) {
this.name = name;
this.age = age;
}
public void setName(String name) {
this.name = name;
}
public String getName() {
return name;
}
public void setAge(int age) {
this.age = age;
}
public int getAge() {
return age;
}
}
ServerImpl.java
import java.rmi.*;
import java.rmi.server.*;
import java.rmi.registry.*;
import javax.naming.*;
public class ServerImpl extends UnicastRemoteObject implements Server {
public ServerImpl()
throws RemoteException {}
public String helloWorld(String name)
throws RemoteException {
return name + "HelloWorld!";
}
public Person getPerson(String name , int age)
throws RemoteException {
return new Person(name , age);
}
public static void main(String[] args)
throws Exception {
Server imp = new ServerImpl();
LocateRegistry.createRegistry(1099);Naming.rebind("rmi://:1099/server", imp);
}
}
Client.java
import javax.naming.*;
import java.rmi.*;
public class RMIClient {
public static void main(String[] args) throws Exception {
Server server = (Server)Naming.lookup("rmi://:1099/server");
System.out.println(server instanceof java.rmi.server.RemoteStub);
System.out.println(server.helloWorld("bruce"));
System.out.println(server.getPerson("bruce",20));
}
}