RMI的概念

RMI(Remote Method Invocation)远程方法调用是一种计算机之间利用远程对象互相调用实现双方通讯的一种通讯机制。使用这种机制,某一台计算机上的对象可以调用另外 一台计算机上的对象来获取远程数据。RMI是Enterprise JavaBeans的支柱,是建立分布式Java应用程序的方便途径。在过去,TCP/IP套接字通讯是远程通讯的主要手段,但此开发方式没有使用面向对 象的方式实现开发,在开发一个如此的通讯机制时往往令程序员感觉到乏味,对此RPC(Remote Procedure Call)应运而生,它使程序员更容易地调用远程程序,但在面对复杂的信息传讯时,RPC依然未能很好的支持,而且RPC未能做到面向对象调用的开发模 式。针对RPC服务遗留的问题,RMI出现在世人面前,它被设计成一种面向对象的通讯方式,允许程序员使用远程对象来实现通信,并且支持多线程的服务,这 是一次远程通讯的革命,为远程通信开辟新的里程碑。

RMI的开发步骤

  1. 先创建远程接口及声明远程方法,注意这是实现双方通讯的接口,需要继承Remote
  2. 开发一个类来实现远程接口及远程方法,值得注意的是实现类需要继承UnicastRemoteObject
  3. 通过javac命令编译文件,通过java -server 命令注册服务,启动远程对象
  4. 最后客户端查找远程对象,并调用远程方法

简单实例

首先为服务建立一个Model层,注意因为此对象需要现实进行远程传输,所以必须继承Serializable

  1. package rmi.model;
  2. import java.io.Serializable;
  3. //注意对象必须继承Serializable
  4. public class PersonEntity implements Serializable {
  5. private int id;
  6. private String name;
  7. private int age;
  8. public void setId(int id) {
  9. this.id = id;
  10. }
  11. public int getId() {
  12. return id;
  13. }
  14. public void setName(String name) {
  15. this.name = name;
  16. }
  17. public String getName() {
  18. return name;
  19. }
  20. public void setAge(int age) {
  21. this.age = age;
  22. }
  23. public int getAge() {
  24. return age;
  25. }
  26. }

创建远程接口PersonService,注意远程接口需要继承Remote

  1. package rmi.service;
  2. import java.rmi.Remote;
  3. import java.rmi.RemoteException;
  4. import java.util.List;
  5. import rmi.model.*;
  6. //此为远程对象调用的接口,必须继承Remote类
  7. public interface PersonService extends Remote {
  8. public List<PersonEntity> GetList() throws RemoteException;
  9. }

建立PersonServiceImpl实现远程接口,注意此为远程对象实现类,需要继承UnicastRemoteObject

  1. package rmi.serviceImpl;
  2. import java.rmi.RemoteException;
  3. import java.rmi.server.UnicastRemoteObject;
  4. import java.util.LinkedList;
  5. import java.util.List;
  6. import rmi.model.PersonEntity;
  7. import rmi.service.*;
  8. //此为远程对象的实现类,须继承UnicastRemoteObject
  9. public class PersonServiceImpl extends UnicastRemoteObject implements PersonService {
  10. public PersonServiceImpl() throws RemoteException {
  11. super();
  12. // TODO Auto-generated constructor stub
  13. }
  14. @Override
  15. public List<PersonEntity> GetList() throws RemoteException {
  16. // TODO Auto-generated method stub
  17. System.out.println("Get Person Start!");
  18. List<PersonEntity> personList=new LinkedList<PersonEntity>();
  19. PersonEntity person1=new PersonEntity();
  20. person1.setAge(25);
  21. person1.setId(0);
  22. person1.setName("Leslie");
  23. personList.add(person1);
  24. PersonEntity person2=new PersonEntity();
  25. person2.setAge(25);
  26. person2.setId(1);
  27. person2.setName("Rose");
  28. personList.add(person2);
  29. return personList;
  30. }
  31. }

建立服务器端,在服务器端注册RMI通讯端口与通讯路径,然后通讯javac命令编译文件,通过java -server 命令注册服务。以下面代码为例,如果阁下将项目建立于D://RMI/RemotingService文件夹上时,则先输入D://RMI /RemotingService/src>javac rmi/remotingservice/Program.java获取Program.class(如何阁下使用的MyEclipse等开发工具,可跳 过此步,直接在*/bin文件夹中直接调用已经生成的Program.class),然后输入D://RMI/RemotingService /src>java rmi/remotingservice/Program启动服务。

  1. package rmi.remotingservice;
  2. import java.rmi.Naming;
  3. import java.rmi.registry.LocateRegistry;
  4. import rmi.service.*;
  5. import rmi.serviceImpl.*;
  6. public class Program{
  7. public static void main(String[] args) {
  8. // TODO Auto-generated method stub
  9. try {
  10. PersonService personService=new PersonServiceImpl();
  11. //注册通讯端口
  12. LocateRegistry.createRegistry(6600);
  13. //注册通讯路径
  14. Naming.rebind("rmi://127.0.0.1:6600/PersonService", personService);
  15. System.out.println("Service Start!");
  16. } catch (Exception e) {
  17. // TODO Auto-generated catch block
  18. e.printStackTrace();
  19. }
  20. }
  21. }

最后建立客户端进行测试,注意客户调用的RMI路径必须服务器配置一致

转自:http://blog.csdn.net/Leslies2/article/details/6436847

  1. package rmi.remotingclient;
  2. import java.rmi.Naming;
  3. import java.util.List;
  4. import rmi.model.PersonEntity;
  5. import rmi.service.*;
  6. public class Program {
  7. public static void main(String[] args){
  8. try{
  9. //调用远程对象,注意RMI路径与接口必须与服务器配置一致
  10. PersonService personService=(PersonService)Naming.lookup("rmi://127.0.0.1:6600/PersonService");
  11. List<PersonEntity> personList=personService.GetList();
  12. for(PersonEntity person:personList){
  13. System.out.println("ID:"+person.getId()+" Age:"+person.getAge()+" Name:"+person.getName());
  14. }
  15. }catch(Exception ex){
  16. ex.printStackTrace();
  17. }
  18. }
  19. }

常见错误

  1. 在命令提示符调用java命令时,显示并无此命令。这是因为未在“环境变量”中绑定JAVA的JDK命令造成的,你首先单击“计算机右 键”->“属性”->“高级”->“环境变量”。在系统变量Path设置中加载为JDK的路径  .;D:/Program Files/Genuitec/Common/binary/com.sun.java.jdk.win32.x86_1.6.0.013/bin。然后 在ClassPath加载服务器端的Program.class地址 .;D://RMI/RemotingService/bin
  2. 在调用javac命令时出现“javac 找不到文件 ..... ”此错误,可能是因为阁下输入的文件路径出现错误造成,注意不要把D://RMI/RemotingService/src>javac rmi/ remotingserviceProgram.java写错为D://RMI/RemotingService/src>javac rmi. remotingservice.Program.java
  3. 在调用D://RMI/RemotingService/bin>java rmi/remotingservice/Program命令时出现“Exception in thread 'main' java.lang.NoClassEdfoundError”错误,第一这可能是阁下把Program 错写为Program.class ,注意java命令不需要加后缀名。第二可能是阁下把“java rmiremotingserviceProgram”错写为“java rmi/ remotingserviceProgram"。

原代码 (下载 )

JAVA 中的RMI是什么的更多相关文章

  1. K:java中的RMI(Remote Method Invocation)

    相关介绍:  RMI全称是Remote Method Invocation,即远程方法调用.它是一种计算机之间利用远程对象互相调用,从而实现双方通讯的一种通讯机制.使用这种机制,某一台计算机(虚拟机) ...

  2. Java基础中的RMI介绍与使用

    今天在这边介绍一下Java基础中的rmi使用.其实rmi有什么样的使用场景呢?它跟webservice有什么区别呢?其实webservice主要是可以跨语言实现项目间的方法调用,而rmi只是java内 ...

  3. 关于<Java 中 RMI、JNDI、LDAP、JRMP、JMX、JMS那些事儿(上)>看后的一些总结-1

    原文地址:https://www.anquanke.com/post/id/194384#h3-3 1.java rmi 关于rmi客户端和服务端通信的过程,java的方法都实现在rmi服务端,客户端 ...

  4. 【译】Java中的对象序列化

    前言 好久没翻译simple java了,睡前来一篇. 译文链接: http://www.programcreek.com/2014/01/java-serialization/ 什么是对象序列化 在 ...

  5. java中serializable

    java中serializable是一个对象序列化的接口,一个类只有实现了Serializable接口,它的对象才是可序列化的.因此如果要序列化某些类的对象,这些类就必须实现Serializable接 ...

  6. java 对象序列化 RMI

    对于一个存在于Java虚拟机中的对象来说,其内部的状态只保持在内存中.JVM停止之后,这些状态就丢失了.在很多情况下,对象的内部状态是需要被持久化下来的.提到持久化,最直接的做法是保存到文件系统或是数 ...

  7. Java中的SerialVersionUID

    Java中的SerialVersionUID 序列化及SergalVersionUID困扰着许多Java开发人员.我经常会看到这样的问题,什么是SerialVersionUID,如果实现了Serial ...

  8. K:java中的安全模型(沙箱机制)

    本博文整合自:Java安全--理解Java沙箱.Java 安全模型介绍.Java的沙箱机制原理入门 相关介绍:  我们都知道,程序员编写一个Java程序,默认的情况下可以访问该机器的任意资源,比如读取 ...

  9. java中使用axis发布和调用webService及dom4j解析xml字符串

    工作中需要调用webService服务,这里记录一下如何在java中发布和调用webService. 需要的jar包: webService服务端: import javax.jws.WebMetho ...

随机推荐

  1. redis安装方法

    redis安装方法1.通过lnmp一键安装包,然后执行./addons.sh install redis2.yum -y install redis3.wget http://redis.google ...

  2. PHP中将对数据库的操作,封装成一个工具类以及学会使用面向对象的方式进行编程

    <?php class SqlTool { //属性 private $conn; private $host="localhost"; private $user=&quo ...

  3. jupyter巨好玩-常用设置

    整理一下有用的设置: 自定义ipython工作目录 用jupyter notebook默认打开的页面时,默认在的是home路径,会看见一大堆东西,假如我们只想展示jupyter的工程目录咋办? 简单说 ...

  4. POJ 2429 GCD & LCM Inverse(Pollard_Rho+dfs)

    [题目链接] http://poj.org/problem?id=2429 [题目大意] 给出最大公约数和最小公倍数,满足要求的x和y,且x+y最小 [题解] 我们发现,(x/gcd)*(y/gcd) ...

  5. 关于iOS中用AudioFile相关API解码或播放AAC_HE_V2时仅仅能识别单声首22.05k採样率的问题

    关于iOS中用AudioFile相关API解码或播放AAC_HE_V2时仅仅能识别单声首22.05k採样率的问题 在官方AQPlayer Demo 和 aqofflinerender中.都用了Audi ...

  6. auto_ptr 要点解析

    今天看了auto_ptr类的用法,又仔细看了看C++标准库中的符合标准的auto_ptr类别的实作,觉得自己基本上理解了auto_ptr的原理,下面就我的心得写几句,有不正确的地方,希望多多指教. 1 ...

  7. HTML5 元素拖放

    <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <title> ...

  8. Ubuntu Code::Blocks IDE 13.12 汉化

    Ubuntu Code::Blocks IDE 13.12 汉化: 安装很简单,不再赘述. 单说汉化: .下载中文简体汉化包(百度网盘):链接: http://pan.baidu.com/s/1kU3 ...

  9. 学习Javascript DOM 编程艺术的一点心得

    最近又看了一遍JS DOM编程艺术,照例来写一写读后感. 其实,我从中学到最深的是几个概念:1.平稳退化.当浏览器并不支持JS的时候网页的基本核心功能是还可以用的:2.逐渐增强.在原始的信息层上用其他 ...

  10. Qt 之 show,hide,setVisible,setHidden,close 等小结

    0QObject::deleteLater()delete obj;析构对象1QWidget::setVisible(bool)使得Widget可见或不可见2QWidget::setHidden(bo ...