JAVA RMI 文档中文(四)

3.4 类的动态加载
RMI允许任意可序列化的对象作为参数、返回值或者异常在RMI调用中传递。RMI利用对象的序列化机制实现虚拟机间对象的传递,也会使用合适的本地信息命名调用流,以便接收端可以下载这个文件。

当远端方法的参数和返回值在接收端的虚拟机中分解组合为对象时,在流中的所有对象类型都需要类定义。分解组合过程中,首先会尝试通过本地类加载上下文中的名字来解析类。RMI也会为动态加载类提供方法,类加载是为了定义那些从远端端点传递过来的参数或者返回值的具体类型的类定义。其中包括了动态下载远端stub类对应的远端对象实现类(通常包括远端引用)和其他任意通过RMI调用传递过来的值,比如一个已经声明的参数类型的子类型,尽管它在接收端的上下文中还是无效的。
RMI also provides a facility for dynamically loading the class definitions for the actual types of objects passed as parameters and return values for remote method invocations from network locations specified by the transmitting endpoint.(这段翻译的不好,原文在这里)

为了支持动态类加载,RMI在运行时使用了java.io.ObjectOutputStreamjava.io.ObjectInputStream的特殊子类来读写RMI的参数和返回值。子类分别重写了ObjectOutputStreamannotateClass方法和ObjectInputStreamresolveClass方法来发送信息,这些信息包含了在数据流中类描述符(指的应该是Class对象)所对应的类文件位置。

对于每一个写在RMI数据流中的类描述符,annotateClass方法都会将该class对象的java.rmi.server.RMIClassLoader.getClassAnnotation方法的调用结果加入到该数据流中,getClassAnnotation的调用结果可能为null,可能为String对象,该String对象存储了远端端点提供的一个URL路径(使用空格分割的URL列表),该路径可以用来下载特定class文件。

对于每一个从RMI数据流中读出来的类描述符,resolveClass方法会从该流中读出一个单独的对象。如果该对象是一个String(而且java.rmi.server.userCodebaseOnly参数的值不为true),那么resolveClass方法返回调用RMIClassLoader.loadClass方法的返回结果,RMIClassLoader.loadClass方法的第一个参数是之前获取到的注解String对象,第二个参数是需要下载的类的类描述符(也就是这个类的Class对象)。如果没有获取到String对象,那么resolveClass方法会返回RMIClassLoader.loadClass方法只有需要下载的类的类描述符作为唯一参数的返回结果。

猜你喜欢

转载自blog.csdn.net/tyn243222791/article/details/82596877