分布式通信框架 - rmi

1)什么是rmi

2)简单的实现rmi

3)rmi原理

4)手写rmi框架

进群:697699179可以获取Java各类入门学习资料!

这是我的微信公众号【编程study】各位大佬有空可以关注下,每天更新Java学习方法,感谢!

学习中遇到问题有不明白的地方,推荐加小编Java学习群:697699179内有视频教程 ,直播课程 ,等学习资料,期待你的加入

首先谈下什么RPC?

Remote procedure call protocal 远程过程调用协议

不用知道具体细节,调用远程系统中类的方法,就跟调用本地方法一样。

RPC协议其实是一种规范。

包括Dubbo,Thrift,rmi,webservice,hessain

网络协议和网络IO对于调用端和服务端来说是透明的。

一个RPC框架应该包含的要素:

RMI概述

rmi(remote method invocation) 远程方法调用

可以认为是RPC的java版本

RMI使用的是JRMP(JAVA Remote Messageing Protocol),可以说JRMP是专门为java定制的通信协议,所以它是纯java的分布式解决方案。

怎么实现一个RMI程序

1)创建远程接口并且继承java.rmi.Remote 接口

packagecom.llf.rmidemo;importjava.rmi.Remote;importjava.rmi.RemoteException;publicinterfaceSayHelloextendsRemote{publicStringsayHello(String name)throwsRemoteException;}复制代码

2)实现我们这个远程接口并且继承 UnicastRemoteObject

packagecom.llf.rmidemo;importjava.rmi.RemoteException;importjava.rmi.server.UnicastRemoteObject;publicclassSayHelloImplextendsUnicastRemoteObjectimplementsSayHello{protectedSayHelloImpl()throwsRemoteException{}@OverridepublicStringsayHello(Stringname)throwsRemoteException{return"Hello LLF -->"+name;}}复制代码

3)创建服务器程序 调用createRegistry方法注册远程对象

package com.llf.rmidemo;importjava.net.MalformedURLException;importjava.rmi.AlreadyBoundException;importjava.rmi.Naming;importjava.rmi.RemoteException;importjava.rmi.registry.LocateRegistry;publicclassHelloServer{publicstaticvoid main(String[] args) {try{SayHellohello=newSayHelloImpl();LocateRegistry.createRegistry(8888);try{Naming.bind("rmi://localhost:8888/sayhello", hello);System.out.println("Server start success!");}catch(MalformedURLExceptione) {e.printStackTrace();}catch(AlreadyBoundExceptione) {e.printStackTrace();}}catch(RemoteExceptione) {e.printStackTrace();}}}复制代码

4)创建客户端程序

package com.llf.rmidemo;importjava.net.MalformedURLException;importjava.rmi.Naming;importjava.rmi.NotBoundException;importjava.rmi.RemoteException;publicclassHelloClient{publicstaticvoid main(String[] args) {try{SayHellohello=(SayHello)Naming.lookup("rmi://localhost:8888/sayhello");System.out.println(hello.sayHello("FXP"));}catch(MalformedURLExceptione) {e.printStackTrace();}catch(RemoteExceptione) {e.printStackTrace();}catch(NotBoundExceptione) {e.printStackTrace();}}}复制代码

结果:

自己去实现一个RMI

1)编写服务器程序,暴露监听,可以使用socket 2)编写客户端程序,通过IP和端口连接到指定的服务,并且把我们的数据做封装(序列化) 3)服务器端收到请求先反序列化在进行业务逻辑处理,把返回结果序列化返回

rmi框架调用时序图

rmi源码分析

我们近乎的可以以如下的图来理解: a) stub和skeleton这俩个身份都是作为代理存在,客户端的称为stub,服务端的称为skeleton,通过这俩个对象屏蔽了远程方法调用的具体细节,这俩个是必不可少的。 b)Registry:注册所,提供了服务名到服务的映射。

结合这上面的图,再以上面的demo代码,我们来扒一扒rmi的底层源码 首先我们看提供服务的server方 createRegistry方法

服务端先创建了一个RegistryImpl的对象,然后做了一个安全校验,这边我们不用关注,重点是看setup方法。

进入到RegistryImpl类

然后进入到UnicastServerRef的exportObject方法 1)首先为传入的RegistryImpl创建一个代理对象stub 2)把UnicastServerRef的skeleton对象设置为当前RegistryImpl对象 3)skeleton,stub,unicastserverRef对象,id和一个boolean构造了一个target对象

再往下追就是export的exportObject方法

主要是调用listen方法创建一个serversocket,启动一条线程等待客户端请求。 至此为止我们的服务端已经起了服务等待客户端连接了。

客户端

这边其实就是创建一个stub的代理对象

用代码来模拟RMI底层过程如下: 新建一个User的对象

packagecom.llf.rmi;publicclassUser{privateintage;publicintgetAge(){returnage;}publicvoidsetAge(intage){this.age = age;}}复制代码

编写一个Skeleton类供客户端调用【这块是rmi定义出来屏蔽底层序列化及流连接的,这边模拟写了下底层的序列化及流】

packagecom.llf.rmi;importjava.io.IOException;importjava.io.ObjectInputStream;importjava.io.ObjectOutputStream;importjava.net.ServerSocket;importjava.net.Socket;//server程序publicclassUser_SkeletonextendsThread{privateUserServeruserServer;publicUser_Skeleton(UserServeruserServer) {this.userServer = userServer;}public void run() {ServerSocketserverSocket =null;ObjectInputStreamread =null;ObjectOutputStreamoos =null;Socketsocket=null;try{serverSocket =newServerSocket(8888);    socket = serverSocket.accept();while(socket !=null) {read =newObjectInputStream(socket.getInputStream());Stringmethod = (String) read.readObject();if(method.equals("age")) {int age = userServer.getAge();oos =newObjectOutputStream(socket.getOutputStream());oos.writeInt(age);oos.flush();}}}catch(Exceptione) {e.printStackTrace();}finally{if(serverSocket !=null) {try{oos.close();read.close();socket.close();serverSocket.close();}catch(IOExceptione) {e.printStackTrace();}}}}}复制代码

写一个stub

packagecom.llf.rmi;importjava.io.IOException;importjava.io.ObjectInputStream;importjava.io.ObjectOutputStream;importjava.net.Socket;importjava.net.UnknownHostException;publicclassUser_StubextendsUser{privateSocketsocket;publicUser_Stub() {try{socket=newSocket("localhost",8888);}catch(UnknownHostExceptione) {e.printStackTrace();}catch(IOExceptione) {e.printStackTrace();}}public int getAge(){ObjectOutputStreamoos=null;ObjectInputStreamois=null;try{oos=newObjectOutputStream(socket.getOutputStream());oos.writeObject("age");oos.flush();ois=newObjectInputStream(socket.getInputStream());returnois.readInt();}catch(IOExceptione) {e.printStackTrace();}finally{try{ois.close();oos.close();}catch(IOExceptione) {// TODO Auto-generated catch blocke.printStackTrace();}}return0;}}复制代码

编写服务器代码

packagecom.llf.rmi;publicclassUserServerextendsUser{public static void main(String[] args) {UserServerserver=newUserServer();server.setAge(18);//模拟rmi生成的skeleton代理对象User_Skeletonuser_Skeleton=newUser_Skeleton(server);user_Skeleton.start();}}复制代码

编写客户端代码

packagecom.llf.rmi;publicclassUserClient{publicstaticvoidmain(String[] args){User user=newUser_Stub();intage=user.getAge();System.out.println(age);}}复制代码

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

推荐阅读更多精彩内容