1. Java RMI介绍

RMI:远程方法调用(Remote Method Invocation)。能够让在某个java虚拟机上的对象像调用本地对象方法一样调用另一个java 虚拟机中的对象上的方法。

1.RMI远程调用步骤

  1. 客户对象调用客户端辅助对象上的方法
  2. 客户端辅助对象打包调用信息(变量,方法名),通过网络发送给服务端辅助对象
  3. 服务端辅助对象将客户端辅助对象发送来的信息解包,找出真正被调用的方法以及该方法所在对象
  4. 调用真正服务对象上的真正方法,并将结果返回给服务端辅助对象
  5. 服务端辅助对象将结果打包,发送给客户端辅助对象
  6. 客户端辅助对象将返回值解包,返回给客户对象
  7. 客户对象获得返回值

2.RMI优缺点

  • 优点: 移植性好,服务器代码可移植到客户端;
  • 缺点: 服务端和客户端都需要是Java。

对于客户对象来说,步骤2-6是完全透明的。

以上简明扼要的介绍了什么是RMI和调用步骤,接下来,说说在程序中是如何使用rmi的。

2. 仅在jdk下程序中的使用

1. RMI服务端

1.创建一个服务端项目rmi-server

2.创建远程方法接口,该接口需要实现Remote接口

public interface IHelloService extends Remote {

    public String sayHello(String name) throws RemoteException;

    public int sum(int a, int b) throws RemoteException;
}

3.创建远程方法接口实现类,然后执行main方法就等于启动了rmi服务端了

public class HelloServiceImpl extends UnicastRemoteObject implements IHelloService {

    public HelloServiceImpl() throws RemoteException {
super();
}
@Override
public String sayHello(String name) throws RemoteException {
return "hello: " + name;
} @Override
public int sum(int a, int b) throws RemoteException {
return a + b;
} public static void main(String args[]) {
try {
//创建一个远程对象
IHelloService helloService = new HelloServiceImpl();
//生成远程对象注册表Registry的实例,并指定端口为8888(默认端口是1099)
LocateRegistry.createRegistry(8888); //把远程对象注册到RMI注册服务器上,并命名为RHello
//绑定的URL标准格式为:rmi://host:port/name(协议名可以省略,下面两种写法都可以)
Naming.bind("rmi://127.0.0.1:8888/HelloService", helloService); System.out.println(">>INFO:远程IHello对象绑定成功!");
} catch (RemoteException e) {
System.out.println("创建远程对象发生异常!");
e.printStackTrace();
} catch (AlreadyBoundException e) {
System.out.println("发生重复绑定对象异常!");
e.printStackTrace();
} catch (MalformedURLException e) {
System.out.println("发生URL畸形异常!");
e.printStackTrace();
}
}
}

2. RMI客户端

1.新建一个客户端项目rmi-client

2.复制服务端的远程方法接口IHelloService,注意,包名要一致

3.测试类调用RMI服务

public class HelloServiceClientTest {
@Test
public void testSayHello() {
try {
// 在RMI服务注册表中查找名称为RHello的对象,并调用其上的方法
IHelloService rhello = (IHelloService) Naming.lookup("rmi://127.0.0.1:8888/HelloService");
System.out.println(rhello.sayHello("world"));
System.out.println(rhello.sum(454, 5457));
} catch (Exception e) {
e.printStackTrace();
}
}
}

2. 在spring封装下程序中的使用

1. RMI服务端

1.创建一个基于Sringboot的项目rmi-server

2.创建一个远程方法接口

package com.wp.learn.spring.remoting;

import java.rmi.Remote;
import java.rmi.RemoteException; public interface IHelloService extends Remote { public String sayHello(String name) throws RemoteException; public int sum(int a, int b) throws RemoteException;
}

3.实现远程方法接口

package com.wp.learn.spring.remoting.impl;

import com.wp.learn.jdk.remoting.IHelloService;
import org.springframework.stereotype.Service; import java.rmi.RemoteException; @Service
public class HelloServiceImpl implements IHelloService { @Override
public String sayHello(String name) throws RemoteException {
try {
Thread.sleep(10000);
} catch (InterruptedException e) {
e.printStackTrace();
}
return "hello: " + name;
} @Override
public int sum(int a, int b) throws RemoteException {
return a + b;
}
}

4.rmi相关配置

@Configuration
public class RMIConfig { @Autowired
@Qualifier("helloServiceImpl")
private HelloServiceImpl helloService; @Bean
public RmiServiceExporter initRmiServiceExporter(){
RmiServiceExporter exporter=new RmiServiceExporter();
exporter.setServiceInterface(IHelloService.class);
exporter.setServiceName("HelloService");
exporter.setService(helloService);
exporter.setRegistryPort(10086);
return exporter;
}
}

2. RMI客户端

1.创建一个基于springboot的项目rmi-client

2.创建和服务端一样的远程方法接口,注意包名一致

3.rmi配置

@Configuration
public class RMIClientConfig { @Bean(name = "helloService")
public RmiProxyFactoryBean initRmiProxyFactoryBean() {
RmiProxyFactoryBean factoryBean = new RmiProxyFactoryBean();
factoryBean.setServiceUrl("rmi://127.0.0.1:10086/HelloService");
factoryBean.setServiceInterface(IHelloService.class);
factoryBean.setLookupStubOnStartup(false);
factoryBean.setRefreshStubOnConnectFailure(true);
// lookupStubOnStartup : 这个属性是表示,不在容器启动的时候创建与Server端的连接;
// refreshStubOnConnectFailure : 这个属性是表示是否连接出错时自动重连;
// registryClientSocketFactory : 这个是客户端与服务端创建SOCKECT的一个工厂。
return factoryBean;
}
}

4.调用接口测试

@RestController
@RequestMapping(value = "/v1/")
public class RMIClientController { // @Autowired
// @Qualifier(value = "helloService")
// IHelloService helloService; @Autowired
IHelloService helloService; @RequestMapping(value = "/test")
public String test1() {
// IHelloService helloService = (IHelloService) factoryBean.getObject();
try {
System.out.println(helloService.sayHello("小青"));
System.out.println("----------------------------------");
System.out.println(helloService.sum(132, 355));
} catch (RemoteException e) {
e.printStackTrace();
}
return "123";
}
}

大功告成!源码地址: https://gitee.com/wangpinggs/learn/tree/master

相关博客:

http://www.blogjava.net/zhenyu33154/articles/320245.html

http://elf8848.iteye.com/blog/1961205

Java RMI使用的更多相关文章

  1. Exception thrown by the agent : java.rmi.server.ExportException: Port already in use

    今天有个应用一直起不来,感觉配置都对啊,奇了怪了.看日志发现如下: STATUS | wrapper | 2017/01/04 08:09:31 | Launching a JVM...INFO | ...

  2. Java RMI之HelloWorld篇

    Java RMI 指的是远程方法调用 (Remote Method Invocation).它是一种机制,能够让在某个 Java 虚拟机上的对象调用另一个 Java 虚拟机中的对象上的方法.可以用此方 ...

  3. 启动tomcat时 错误: 代理抛出异常 : java.rmi.server.ExportException: Port already in use: 1099;

     错误: 代理抛出异常 : java.rmi.server.ExportException: Port already in use: 1099; nested exception is:  java ...

  4. JAVA RMI 实例

    下面我将介绍一个完整的实例,让初学者能快速体验RMI的功用. 分为以下四个步骤 1. 创建远程接口及声明远程方法(HelloInterface.java)2. 实现远程接口及远程方法(继承Unicas ...

  5. JAVA RMI例子

    RMI 是java语言的一个RPC框架,本文给出基础例子如下: 1.实现接口: public interface ICalc extends Remote { public int add(int p ...

  6. java.rmi.NoSuchObjectException: no such object in table

    jmx链接的时候,最简单的例子都行不通,郁闷,出现了: 参考:http://reiz6153.blog.163.com/blog/static/401089152009442723208/ 代码: M ...

  7. 启动tomcat时 错误: 代理抛出异常 : java.rmi.server.ExportException: Port already in use: 1099的解决办法

    一.问题描述 今天一来公司,在IntelliJ IDEA 中启动Tomcat服务器时就出现了如下图所示的错误:

  8. Java RMI 介绍和例子以及Spring对RMI支持的实际应用实例

    RMI 相关知识 RMI全称是Remote Method Invocation-远程方法调用,Java RMI在JDK1.1中实现的,其威力就体现在它强大的开发分布式网络应用的能力上,是纯Java的网 ...

  9. Java学习笔记(十六)——Java RMI

    [前面的话] 最近过的好舒服,每天过的感觉很充实,一些生活和工作的技巧注意了就会发现,其实生活也是可以过的如此的有滋有味,满足现在的状况,并且感觉很幸福. 学习java RMI的原因是最近在使用dub ...

  10. JAVA RMI helloworld入门

    Java RMI 指的是远程方法调用 (Remote Method Invocation).它是一种机制,能够让在某个 Java 虚拟机上的对象调用另一个 Java 虚拟机中的对象上的方法.可以用此方 ...

随机推荐

  1. 关于字符串math函数的用法例子

    var objStr=new String("Yue I love you till the end of my life!"); var reg3 = /[^\s+]/g; ob ...

  2. angularjs之ui-bootstrap的Datepicker Popup不使用JS实现双日期选择控件

    最开始使用ui-bootstrap的Datepicker Popup日期选择插件实现双日期选择时间范围时,在网上搜了一些通过JS去实现的方法,不过后来发现可以不必通过JS去处理,只需要使用其自身的属性 ...

  3. zabbix3.4.7搭建及邮件告警

    Zabbix3.4.7部署 系统环境:CentOs7.2 1.关闭selinux 1.1 [root@localhost ~]# setenforce 0 #临时关闭 1.2 [root@localh ...

  4. ElasticSearch 5.2.2 安装及 head 插件的安装

    ElasticSearch 是一个基于 Lucene 的高度可扩展的开源全文搜索和分析引擎.它能够做到可以快速.实时地存储.搜索和分析大量数据.它通常作为底层引擎/技术,为具有复杂搜索功能和要求的应用 ...

  5. OA项目笔记

    一.创建项目构架 1.创建一个Maven的web工程 1.1修改编译器版本 <properties> <project.build.sourceEncoding>UTF-8&l ...

  6. Springboot2.0访问Redis集群

    Redis 是一个开源(BSD许可)的,内存中的数据结构存储系统,它可以用作高性能的key-value数据库.缓存和消息中间件,掌握它是程序员的必备技能,下面是一个springboot访问redis的 ...

  7. 洛谷——P2695 骑士的工作

    https://www.luogu.org/problem/show?pid=2695 题目背景 你作为一个村的村长,保卫村庄是理所当然的了.今天,村庄里来了一只恶龙,他有n个头,恶龙到处杀人放火.你 ...

  8. 洛谷——P1525 关押罪犯

    https://www.luogu.org/problem/show?pid=1525 题目描述 S 城现有两座监狱,一共关押着N 名罪犯,编号分别为1~N.他们之间的关系自然也极不和谐.很多罪犯之间 ...

  9. RGB 颜色空间转 HSI 颜色空间的matlab程序实现

    RGB 颜色空间转 HSI 颜色空间的matlab程序实现 2014.10.20之前的内容有误,这里依据wikipedia更新了算法内容. 算法以wiki为准 https://en.wikipedia ...

  10. 堆-heap

    #include <vector> #include <cstdio> using namespace std; class Heap { private : vector&l ...