Chinaunix首页 | 论坛 | 博客
  • 博客访问: 1675943
  • 博文数量: 585
  • 博客积分: 14610
  • 博客等级: 上将
  • 技术积分: 7402
  • 用 户 组: 普通用户
  • 注册时间: 2008-05-15 10:52
文章存档

2013年(5)

2012年(214)

2011年(56)

2010年(66)

2009年(44)

2008年(200)

分类:

2010-11-29 02:02:09

Java2 RMI 入门 


RMI从Java1.1开始,RMI使得运行于不同JVM(包括不同主机)上的Java应用程序可以彼此通话。 
即:一个JVM中的Java应用程序可以调用另一JVM上的对象(远程对象)所定义的方法。 
Java RMI有着重要的意义。RMI在Java网络编程和高级编程中都有重要的应用,如EJB, Jini等。 
Java2对RMI做了很多增强和改进,如安全性,动态代码下载等。 
本文给出了一个最简单的例子,以说明其中的一些基本原理。本文的特点是注重了实际开发和真正的运行 
环境相结合,模拟了RMI真正的开发和运行过程。 

1. 实现远程接口,生成远程对象,存根(Stub)和框架(Skeleton) 

实现远程接口,远程接口告诉JVM:实现了该接口的对象可以远程调用及有哪些方法可以调用。 
本例子中定义了sayHello()。由于远程调用会涉及到网络通讯,因此这些方法都要抛出RemoteException. 
远程接口和远程对象可以由A开发,并把远程接口(Hello)d打包分给Client端开发者B。 

建立f:\server目录,把Hello.java和HelloImpl.java拷贝到该目录中。 

// Hello.java 
package jdeveloper.rmi; 

import java.rmi.Remote; 
import java.rmi.RemoteException; 

public interface Hello extends Remote { 
String sayHello() throws RemoteException; 


生成远程对象. 
// HelloImpl.java 
package jdeveloper.rmi; 

import java.rmi.Naming; 
import java.rmi.RemoteException; 
import java.rmi.RMISecurityManager; 
import java.rmi.server.UnicastRemoteObject; 

public class HelloImpl extends UnicastRemoteObject 
implements Hello { 

public HelloImpl() throws RemoteException { 
super(); 


public String sayHello() { 
return "Hello World!"; 


public static void main(String args[]) { 

// Create and install a security manager 
if (System.getSecurityManager() == null) { 
System.setSecurityManager(new RMISecurityManager()); 

try { 
Hello obj = new HelloImpl(); 
// Bind this object instance to the name "HelloServer" 
Naming.rebind("HelloServer", obj); 
System.out.println("HelloServer bound in registry"); 
} catch (Exception e) { 
System.out.println("HelloImpl err: " + e.getMessage()); 
e.printStackTrace(); 



存根(Stub)和框架(Skeleton) 
f: 
cd \server 
javac -d . Hello.java 
javac -d . HelloImpl.java 
rmic -d . jdeveloper.rmi.HelloImpl 

jar cvf hello.jar jdeveloper\rmi\Hello.class 把hello.jar分发给Client端的开发者B。 

存根(Stub)的存放! 
存根(Stub)是动态下载的。Client通过存根(Stub)和远程对象的框架(Skeleton)通讯,对Client来 
讲就象操作本地对象一样。在大多数情况下,可下载的代码放到http服务器的某个目录中。本例子放到下。 
hjc:机器名,rmi:http的一个目录。如果只是单机测试则可以放到某个目录下 如f:\serverclasses. 
本文将对以上两种方法都给出运行的步骤(policy文件和bat文件)。 

2. 实现Client端程序 
// HelloClient.java 
package jdeveloper.rmi; 
import java.rmi.RMISecurityManager; 
import java.rmi.Naming; 
import java.rmi.RemoteException; 
import java.rmi.NotBoundException; 

public class HelloClient { 
public static void main(String args[]) throws Exception{ 
System.setSecurityManager(new RMISecurityManager()); 
Hello RemoteObj = (Hello)Naming.lookup("//"+ args[0] +"/HelloServer"); 
System.out.println(RemoteObj.sayHello()); 



建立f:\client目录,把HelloClient.java拷贝到该目录中。 
建立f:\clientclasses目录,把hello.jar拷贝到该目录中。 
f: 
cd \client 
javac -classpath %CLASSPATH%;f:\clientclasses\hello.jar -d . HelloClient.java 

3. 运行程序 
启动DOS窗口 
set classpath= 
start rmiregistry 

A.以单机方式运行 
建立f:\serverclasses\jdeveloper\rmi目录 
f: 
cd f:\serverclasses 
copy f:\server\hello.jar . 
copy f:\server\jdeveloper\rmi\HelloImpl_Stub.class f:\serverclasses\jdeveloper\rmi\ 
jar xvf hello.jar 

启动新的DOS窗口 
把 starthelloserver.bat 和 rmiserver.policy 放到f:\server\ 
运行 starthelloserver 

starthelloserver.bat 
SET CP=f:\server;f:\serverclasses\hello.jar 
echo using classpath: %CP% 
java -classpath %CP% -Djava.rmi.server.codebase=file:/f:\serverclasses/ 
-Djava.rmi.server.hostname=hjc -Djava.security.policy=rmiserver.policy jdeveloper.rmi.HelloImpl 

rmiserver.policy 
grant { 
permission java.net.SocketPermission "*:1024-65535", 
"connect"; 
}; 

启动新的DOS窗口 
把 starthelloclient.bat 和 rmiclient.policy 放到f:\client\ 
运行 starthelloclient 

starthelloclient.bat 
@echo off 
set CP=f:\client;f:\clientclasses\hello.jar 
echo using classpath %CP% 
@echo on 
java -classpath %CP% -Djava.rmi.server.codebase=file:/f:\serverclasses/ 
-Djava.security.policy=rmiclient.policy jdeveloper.rmi.HelloClient %1 
rmiclient.policy 
grant { 
permission java.net.SocketPermission "*:1024-65535", 
"connect"; 
permission java.io.FilePermission 
"f:\\serverclasses\\-", "read"; 
}; 

B.以网络方式运行 
建立 apache_path\htdocs\rmi\jdeveloper\rmi目录 
cd apache_path\htdocs\rmi 
copy f:\server\hello.jar . 
copy f:\server\jdeveloper\rmi\HelloImpl_Stub.class apache_path\htdocs\rmi\jdeveloper\rmi 
jar xvf hello.jar 
把 starthellohttpserver.bat 和 rmihttpserver.policy 放到f:\server\ 

启动新的DOS窗口 
运行 starthellohttpserver 
starthellohttpserver.bat 
SET CP=f:\server;f:\serverclasses\hello.jar 
echo using classpath: %CP% 
java -classpath %CP% -Djava.rmi.server.codebase=/ 
-Djava.security.policy=rmihttpserver.policy jdeveloper.rmi.HelloImpl 
rmihttpserver.policy 
grant { 
permission java.net.SocketPermission "*:1099", "accept, connect, listen, resolve"; 
permission java.net.SocketPermission "*:80", "accept, connect, listen, resolve"; 
}; 

启动新的DOS窗口 
把 starthellohttpclient.bat 和 rmihttpclient.policy 放到f:\client\ 
运行 starthellohttpclient 
starthellohttpclient.bat 
@echo off 
set CP=f:\client;f:\clientclasses\hello.jar 
echo using classpath %CP% 
@echo on 
java -classpath %CP% -Djava.rmi.server.codebase=/ 
-Djava.security.policy=rmihttpclient.policy jdeveloper.rmi.HelloClient %1 

rmihttpclient.policy 
grant { 
permission java.net.SocketPermission "*:80", "connect"; 
permission java.net.SocketPermission "*:1024-65535", "connect"; 
}; 
阅读(1175) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~