远程过程调用详细教程


一、前言


实验目的

利用RMI技术对远程文件夹进行控制:可以增加文件(文本文件)、修改文件(文本文件)、删除文件、列出文件;统计该文件夹具有多少个文件、占用磁盘空间的总大小。

开发环境
  1. 操作系统:Windows 10 X64
  2. IDE:Intellij IDEA

二、准备

新建Java项目:File->New->Project



点击Next:



点击Next:

设置项目名称以及项目存储位置,点击Finish:


三、代码步骤


1. 远程接口定义

定义用于远程对象的接口 FileService。这个接口定义了客户机能够远程地调用的方法。远程接口和本地接口的主要差异在于,远程方法必须能抛出 RemoteException。具体方法如下:
FileService.java:

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

public interface FileService extends Remote {
    //列出文件
    public String[] play() throws RemoteException;

    //增加文件
    public boolean addFile(String name) throws RemoteException;

    //删除文件
    public boolean deleteFile(String name) throws RemoteException;

    //修改文件
    public boolean alterFile(String name,String content) throws RemoteException;

    //读取文件内容
    public String readFile(String name) throws RemoteException;

    //统计文件信息
    public long[] fileInformation() throws RemoteException;
}

这些方法必须能抛出RemoteException,如果客户机和服务器之间的通信错误,则客户机将捕获此异常。

注:该接口本身继承了java.rmi包中定义的Remote接口。Remote接口本身没有定义方法,但通过继承它,我们说明该接口可以被远程地调用。

2. 实现远程接口

编写一个实现FileService接口的类 FileServiceImpl。
FileServiceImpl.java:

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

/**
 * 继承自UnicastRemoteObject,为远程对象的实现类
 */
public class FileServiceImpl extends UnicastRemoteObject implements FileService{

    public FileServiceImpl() throws RemoteException {
        super();
    }

    // 列出文件
    @Override
    public String[] play() throws RemoteException {
        File file = new File("D://RMI");
        String[] names=file.list();
        return names;
    }

    // 增加文件
    @Override
    public boolean addFile(String name) throws RemoteException {
        File file = new File("D://RMI",name);
        if(!file.exists()){
            try{
                file.createNewFile();
                return true;
            }catch(IOException e){
                e.printStackTrace();
            }
        }
        return false;
    }

    // 删除文件
    @Override
    public boolean deleteFile(String name) throws RemoteException {
        File file = new File("D://RMI",name);
        if(file.exists()){
            file.delete();
            return true;
        }
        return false;
    }

    // 修改文件
    @Override
    public boolean alterFile(String name, String content) throws RemoteException {
        FileWriter fw = null;

        try{
            fw = new FileWriter("D://RMI/"+name);
        }catch(IOException e1){
            e1.printStackTrace();
        }

        try{
            fw.write(content,0,content.length());
        }catch (IOException e2){
            e2.printStackTrace();
        }

        //关闭流
        try{
            fw.close();
            return true;
        }catch (IOException e3){
            e3.printStackTrace();
        }

        return false;
    }

    //读取文件内容
    @Override
    public String readFile(String name) throws RemoteException {
        FileReader fr = null;

        try{
            fr = new FileReader("D://RMI/"+name);
        }catch (FileNotFoundException e){
            e.printStackTrace();
        }

        int ch = 0;
        String filecontent = "";
        try{
            while((ch = fr.read())!=-1){
                filecontent = filecontent + (char)ch;
            }
        }catch (IOException e){
            e.printStackTrace();
        }

        //
        try{
            fr.close();
        }catch (Exception e){
            e.printStackTrace();
        }

        return filecontent;
    }

    //统计该文件夹具有多少个文件、占用磁盘空间的总大小
    @Override
    public long[] fileInformation() throws RemoteException {
        File file = new File("D://RMI");
        File[] files = file.listFiles();
        long number = files.length;
        long size = 0;
        for(int i=0;i<number;i++){
            size = size + files[i].length()/1024;
        }
        long[] information = {number,size};
        return information;
    }
}

注:在对文件进行读写操作后必须关闭用close()函数关闭流,不然可能会影响到delete操作,导致不能删除文件。

3. 编写在服务器上运行的主程序 Server.java。

创建服务器对象的初始实例,然后将对象的名称写到RMI命名注册表。

//创建服务器对象实例
FileService fileService = new FileServiceImpl();
 //注册服务的端口
LocateRegistry.createRegistry(6600);
//绑定本地地址和服务器的路径
Naming.rebind("rmi://127.0.0.1:6600/FileService",fileService);

RMI命名注册表允许您将URL名称分配给对象以便客户机查找它们。要注册名称,需要调用在Naming类上定义的静态rebind方法。这个方法接受对象的URL名称以及对象引用。

名称字符串包含 rmi:// 前缀、运行RMI对象的服务器的计算机主机名和对象本身的名称。这里用了本机地址127.0.0.1。

完整代码如下:
Server.java:

import java.rmi.Naming;
import java.rmi.registry.LocateRegistry;

    public class Server {

        public Server(){}

        public static void main(String[] args){

            try{
                FileService fileService = new FileServiceImpl();
                //注册服务的端口
                LocateRegistry.createRegistry(6600);
                //绑定本地地址和服务器的路径
                Naming.rebind("rmi://192.168.1.102:6600/FileService",fileService);
                System.out.println("开始服务!");
            }catch(Exception e){
                e.printStackTrace();
            }
        }
}
4. 编写客户端程序 Client.java。

使用RMI注册表查找远程对象。

//调用远程对象,RMI路径与接口必须与服务器配置一致
FileService fileService = (FileService)Naming.lookup("rmi://127.0.0.1:6600/FileService");

根据选择的操作,调用有远程接口定义的方法。完整代码如下:
Client.java:

import java.rmi.Naming;
import java.util.Scanner;

public class Client {

    public Client(){}

    public static void main(String[] args){
        try{
            //调用远程对象,RMI路径与接口必须与服务器配置一致
            FileService fileService = (FileService)Naming.lookup("rmi://127.0.0.1:6600/FileService");

            int choose;
            Scanner input = new Scanner(System.in);

            System.out.println("1.列出文件");
            System.out.println("2.增加文件");
            System.out.println("3.修改文件");
            System.out.println("4.删除文件");
            System.out.println("5.统计文件");

            while(true){
                System.out.println("请选择要进行的操作:");
                choose = input.nextInt();
                if(choose == 1){
                    String[] filename = fileService.play();
                    for(int i = 0;i < filename.length;i++){
                        System.out.println(filename[i]);
                    }
                }
                else if(choose == 2){
                    System.out.println("请输入要新增文件的名称:");
                    String name = input.next();
                    boolean flag = fileService.addFile(name);
                    if(flag == true){
                        System.out.println("创建成功!");
                    }
                    else{
                        System.out.println("创建失败!");
                    }
                }
                else if(choose == 3){
                    System.out.println("请输入要修改文件的名称:");
                    String name = input.next();
                    String filecontent = fileService.readFile(name);
                    System.out.println(filecontent);
                    System.out.println("请输入修改内容:");
                    String content = input.next();
                    boolean flag = fileService.alterFile(name,content);
                    if(flag == true){
                        System.out.println("修改成功!");
                    }
                    else{
                        System.out.println("修改失败!");
                    }
                }
                else if(choose == 4){
                    System.out.println("请输入要删除文件的名称:");
                    String name = input.next();
                    boolean flag = fileService.deleteFile(name);
                    if(flag == true){
                        System.out.println("删除成功!");
                    }
                    else{
                        System.out.println("删除失败!");
                    }
                }
                else if(choose == 5){
                    long[] information = fileService.fileInformation();
                    System.out.println("总共有文件夹:"+information[0]+"个.");
                    System.out.println("大小总计:"+information[1]+"KB.");
                }
                else{
                    System.out.println("无效操作!");
                }
            }
        }catch (Exception ex){
            ex.printStackTrace();
        }
    }
}

四、运行

1. 打开Server.java,点击Run,选择Run,在弹出框中选择Server。


运行结果:
2. 打开Client.java,点击Run,选择Run,在弹出框中选择Client。同上。
运行结果:

输入1,显示所有文件:



输入2,新增文件;输入文件名:



输入3,修改文件;输入文件名:

输入4,删除文件;输入文件名:

输入5,显示文件个数和所占磁盘大小:


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

推荐阅读更多精彩内容

  • JAVA面试题 1、作用域public,private,protected,以及不写时的区别答:区别如下:作用域 ...
    JA尐白阅读 1,154评论 1 0
  • 小编费力收集:给你想要的面试集合 1.C++或Java中的异常处理机制的简单原理和应用。 当JAVA程序违反了JA...
    八爷君阅读 4,592评论 1 114
  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,103评论 1 32
  • JAVA相关基础知识 1、面向对象的特征有哪些方面 1.抽象: 抽象就是忽略一个主题中与当前目标无关的那些方面,以...
    yangkg阅读 666评论 0 1
  • 错误信息提示如下: Failed to parse Date value '2018-10-31 13:04:16...
    良人与我阅读 1,149评论 1 2