java.rmi.server.codebase Property介绍

在使用RMI编程的时候,有时会出现如下的错误:

 java.rmi.ServerException: RemoteException occurred in server thread; nested excption is:
        java.rmi.UnmarshalException: error unmarshalling arguments; nested excetion is:
        java.lang.ClassNotFoundException: cn.ac.ict.hla.rmi.HelloImpl_Stub
        at sun.rmi.server.UnicastServerRef.oldDispatch(UnicastServerRef.java:39)
        at sun.rmi.server.UnicastServerRef.dispatch(UnicastServerRef.java:250)
        at sun.rmi.transport.Transport$1.run(Transport.java:159)
        at java.security.AccessController.doPrivileged(Native Method)

开始的时候以为在运行RMI Server部分的lauch Code时找不到类,即使指定了classpath还是不行。

其实不是lauch code找不到类定义,而是rmiregistry找不到类定义,这时有两种解决办法

1.  在类所在目录运行rmiregistry;

2. 在运行lauch code的时候使用java.rmi.server.codebase指定类路径;

第1种方案是有局限的,因为不可能所有的类都在一个目录下,所以比较好的方式还是使用第2种方法。不过在使用第2种方法的使用要注意格式问题,例如:

-Djava.rmi.server.codebase=http://YourServerName/YourPackagePathName/

-Djava.rmi.server.codebase=http://webline/public/mystuff.jar


 -Djava.rmi.server.codebase=file:/d:/SomeSource/Testproject/java/rmi/Hello2/ 

使用本地文件目录时必须使用绝对路径,而且所有的codebase后面都需要有个 /  或 /   来表示这是一个目录,也可以是一个jar文件。如果去掉它们就会报错。

A codebase can be defined as a source, or a place, from which to load classes into a virtual machine.

我们可以把CLASSPATH看作是local codebase,而RMI和applet中使用的是remote codebase。

Applet中codebase的使用方式:

1. 客户浏览器请求一个不做客户CLASSPATH中的类

2. applet的类定义被客户从服务器上使用HTTP下载下来

3. applet在客户机上执行

java.rmi.server.codebase Property介绍_第1张图片

RMI中codebase的使用方式:

java.rmi.server.codebase Property介绍_第2张图片

1.  远程对象服务器通过设定java.rmi.server.codebase 属性指定远程对象的codebase。RMI server向RMI registry注册了一个远程对象,绑定到一个name上,在server VM上设置的codebase被通知给了RMI registry;
2.  RMI client请求一个特定名字的远程对象的引用,用来调用远程方法;
3.  RMI registry返回一个到被请求类的引用。如果这个stub实例的类定义可以在client的CLASSPATH中找到,Client会本地加载这个类;如果client的CLASSPATH中没有类定义,client会尽量从远程对象的codebase中获取类定义;
4.  client从codebase中请求类定义。
5.  对请求的stub的类定义被下载到client;
6.  这时,client就有了调用远程方法的所有信息。这个stub成为一个对Server上remote对象的代理,与Applet不同的是,Applet在本地执行,而RMI在remote VM中执行;

注意:在server向rmiregistry注册的时候,也会产生第4步和第5步,不过请求的角色不是client,而是rmiregistry

codebase不但可以用于加载stub的类定义,也可以用来加载任何其他的类定义,当远程方法的一个参数或者返回值对象的类定义在本地 CLASSPATH中找不到时,也可以使用codebase来加载,如图示:

 客户发起一个包含Remote Server不知道的类的实例的远程调用,Remote从codebase中下载该类的定义。

注:一个典型的情况是一个方法的参数是个Interface,但实际传人的参数是某个实现了这个接口的类,这时即使Remote没有这个类定义,编译也可以通过,但是具体调用时会找不到类定义。

注:这种情况下的codebase要在client端指定,如果不指定,Server就会错用自己的codebase。

 

 

你可能感兴趣的:(thread,server,服务器,interface,applet,nested)