Java RMI使用方式与技巧总结

RMI介绍

RMI全名Remote Method Invocation,用于java程序远程方法(不存在于本地中的方法)调用。是java实现分布式的实现方案之一。它可以被看作是RPC的Java版本。但是传统RPC并不能很好地应用于分布式对象系统。而Java RMI 则支持存储于不同地址空间的程序级对象之间彼此进行通信,实现远程对象之间的无缝远程调用。

RMI开发使用步骤:

RMI将一个基础的Java类通过实现相应的方式,使其可以被其它的外部项目使用。步骤很简单:

  • 步骤一
    制作远程接口。远程接口其实就是本地的一个需要RMI转化的普通接口。提供客户端需要的服务。这个远程接口必须要继承接口Remote

  • 步骤二
    实现远程接口在本地创建其远程接口的实现类。此处可以通过继承JDK中的超类UnicastRemoteObject,进行实现,跳过一些复杂的实现内容。此步骤需要考虑一下异常的处理。实现超类后其构造函数的异常需要捕获。

  • 步骤三
    开启注册。使用指令rmiregistry。

  • 步骤四

启动本地(服务端)的程序,生成对应的target编译文件。注意:第三步与第四步不可逆。不开启服务的情况下,注册是会失败的。导致远程(客户端)调用服务程序出现异常。

  • 步骤五

执行指令rmic +你对应的类的全名。需要注意的是,rmic指令执行的时候,需要把文件夹定位到其target的source下,对于我的尝试项目来说,就是编译后的classes文件夹下,不同的人可能会有不一样的编译路径,同时文件名必须用全名否则其编译指令将会报错找不到对应的文件

  • 步骤六
    为需要调用的远程提供服务端生成的文件。对于步骤五在执行过后,默认的情况下会在服务端的目录下生成一份.class文件。将这份class文件复制到对应的远程(客户端)下即可让客户端调用远程的服务了。

我的项目结构及代码

服务端需要处理的:

package com.model.design.agency;

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

public class MyRemoteImpl extends UnicastRemoteObject implements MyRemote {

    public MyRemoteImpl() throws RemoteException {
    }

    public String doNothing() throws RemoteException {
        return "hasaki";
    }

    public String doSomething() throws RemoteException {
        return "hasai";
    }
}
package com.model.design.agency;

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

public interface MyRemote extends Remote {
    String doNothing() throws RemoteException;
    String doSomething() throws RemoteException;
}
package com.model.design.agency;

import java.net.MalformedURLException;
import java.rmi.AlreadyBoundException;
import java.rmi.Naming;
import java.rmi.RemoteException;
import java.rmi.registry.LocateRegistry;

public class TestClass {
    public static void main(String[] args) {
        try {
            MyRemote remote=new MyRemoteImpl();
            LocateRegistry.createRegistry(1100);
            Naming.rebind("rmi://127.0.0.1:1100/RHello",remote);
        } catch (RemoteException e) {
            e.printStackTrace();
        } catch (MalformedURLException e) {
            e.printStackTrace();
        }
        System.out.println("success");
    }
}

服务端的编译路径:


编译结构.png

客户端需要做的:

目录结构.png

其中MyRemote接口值与服务端内容一致。
服务端编译后生成的—Stub(在RMI中被称为桩的东西,此外还有被称为骨架的存在,但是因为版本迭代原因骨架貌似是被摒弃了)文件:


编译文件路径地址.png

编译后生成的文件内容如下:

//
// Source code recreated from a .class file by IntelliJ IDEA
// (powered by Fernflower decompiler)
//

package com.model.design.agency;

import java.lang.reflect.Method;
import java.rmi.Remote;
import java.rmi.RemoteException;
import java.rmi.UnexpectedException;
import java.rmi.server.RemoteRef;
import java.rmi.server.RemoteStub;

public final class MyRemoteImpl_Stub extends RemoteStub implements MyRemote, Remote {
    private static final long serialVersionUID = 2L;
    private static Method $method_doNothing_0;
    private static Method $method_doSomething_1;

    static {
        try {
            $method_doNothing_0 = (class$com$model$design$agency$MyRemote != null ? class$com$model$design$agency$MyRemote : (class$com$model$design$agency$MyRemote = class$("com.model.design.agency.MyRemote"))).getMethod("doNothing");
            $method_doSomething_1 = (class$com$model$design$agency$MyRemote != null ? class$com$model$design$agency$MyRemote : (class$com$model$design$agency$MyRemote = class$("com.model.design.agency.MyRemote"))).getMethod("doSomething");
        } catch (NoSuchMethodException var0) {
            throw new NoSuchMethodError("stub class initialization failed");
        }
    }

    public MyRemoteImpl_Stub(RemoteRef var1) {
        super(var1);
    }

    public String doNothing() throws RemoteException {
        try {
            Object var1 = super.ref.invoke(this, $method_doNothing_0, (Object[])null, 3201213088384021920L);
            return (String)var1;
        } catch (RuntimeException var2) {
            throw var2;
        } catch (RemoteException var3) {
            throw var3;
        } catch (Exception var4) {
            throw new UnexpectedException("undeclared checked exception", var4);
        }
    }

    public String doSomething() throws RemoteException {
        try {
            Object var1 = super.ref.invoke(this, $method_doSomething_1, (Object[])null, -5449258171918619596L);
            return (String)var1;
        } catch (RuntimeException var2) {
            throw var2;
        } catch (RemoteException var3) {
            throw var3;
        } catch (Exception var4) {
            throw new UnexpectedException("undeclared checked exception", var4);
        }
    }
}

这个不是一个java文件,并不符合java的编程语法,使用的时候将其放置在对应的编译,路径之下就好了。需要java文件的话,可以使用rmic -keep指令保存生成的java文件。默认情况下,这个生成的java文件是作为一个临时文件存在的,会被删除,使用keep指令就可以保证其不被删除了
客户端调用

package com.intretech.ums.mailsystem.controller;

import com.model.design.agency.MyRemote;
import com.model.design.agency.MyRemoteImpl;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

import java.rmi.RemoteException;

@RestController
public class RMIController {
    @RequestMapping("/doSomething")
    public String doSomething(){
        String some=null;
        MyRemote remote=new MyRemoteImpl();
        try {
            some=remote.doSomething();
        } catch (RemoteException e) {
            e.printStackTrace();
        }
        return some;
    }
}

至于rmic指令只要你正确配置了jdk跟jre的环境变量路径就可以使用了,如果使用指令失败,请再三确认一下路径是否正常。transient关键字可以限制将一些对象不进行网络传输

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,686评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,668评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,160评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,736评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,847评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,043评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,129评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,872评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,318评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,645评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,777评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,861评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,589评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,687评论 2 351

推荐阅读更多精彩内容

  • 面向对象编程(OOP) Java是一个支持并发、基于类和面向对象的计算机编程语言。下面列出了面向对象软件开发的优点...
    大家请叫我小杰阅读 1,127评论 0 0
  • 面向对象编程(OOP) Java是一个支持并发、基于类和面向对象的计算机编程语言。下面列出了面向对象软件开发的优点...
    hutuxiaogui阅读 936评论 0 2
  • Java是一个支持并发、基于类和面向对象的计算机编程语言。下面列出了面向对象软件开发的优点: 代码开发模块化,更易...
    安安静静写代码阅读 1,092评论 0 8
  • 大家好,我是IT修真院北京分院第31期的学员,一枚正直纯洁善良的JAVA程序员。今天给大家分享一下,修真...
    ve追风_685b阅读 2,963评论 0 1
  • JAVA面试题 1、作用域public,private,protected,以及不写时的区别答:区别如下:作用域 ...
    JA尐白阅读 1,146评论 1 0