多线程设计模式——承诺模式

场景描述

我们将要模拟指定一个文件目录,将该目录的所有所有的文件上传到ftp服务器上

我们知道获取ftp连接与读取目录中所有文件都是比较耗时的,如果可以同时进行是最好的,我们可以使用多线程来获取连接,但两个线程哪个快无法保证,如何保证当连接创建成功后才开始执行上传文件呢,就可以使用承诺模式


一. FTPConnectionUtil

调用者可以通过getPromise()方法获取Ftp连接承诺者对象FutureTask<FTPConnectionUtil>,我们假设获取连接时间为5秒

public class FTPConnectionUtil {

    private Exception exception;

    private FTPConnectionUtil(){
        //防止通过暴力反射破坏单例
        if(InFTPConnectionUtil.connectionUtil != null){
            throw  new RuntimeException("单例已经存在!");
        }
    }

    public static FTPConnectionUtil newInstance(){
        return InFTPConnectionUtil.connectionUtil;
    }

    /**
     * 内部类单例,只有调用时才会加载,既避免了懒汉式浪费资源,又避免了饿汉式安全问题
     */
    private static class InFTPConnectionUtil{
        private final static FTPConnectionUtil connectionUtil = new FTPConnectionUtil();
    }

    /**
     * 构造方法私有,根据此方法获取FutureTask承诺,通过多线程建立连接,然后通过FutureTask.get()方法获取泛型<T>,
     * 可以通过isDone方法判断多线程是否执行完毕
     * 可以在泛型实体里接受异常属性,通过获取属性判断建立连接过程是否正常</>
     * @return
     */
    public static FutureTask<FTPConnectionUtil> getPromise(){

        Callable<FTPConnectionUtil> callable = new Callable() {
            @Override
            public Object call() throws Exception {
                FTPConnectionUtil ftpConnectionUtil = FTPConnectionUtil.newInstance();
                return ftpConnectionUtil.doFtpConnection(ftpConnectionUtil);
            }
        };

        FutureTask<FTPConnectionUtil> futureTask = new FutureTask<>(callable);
        new Thread(futureTask).start();
        return futureTask;

    }

    private  FTPConnectionUtil doFtpConnection(FTPConnectionUtil util){
        try {
            System.out.println("---------建立连接开始");
            for (int i = 1; i <= 5; i++) {
                Thread.sleep(1000);
                System.out.println("-----建立连接"+i+"秒!");
            }
            System.out.println("---------建立连接完成");
        }catch (Exception e){
            util.exception = e;
        }
        return util;
    }

    public void upload(File file){

        System.out.println(file.getName() + "上传完成!");
    }

    public Exception getException() {
        return exception;
    }

    private void setException(Exception exception) {
        this.exception = exception;
    }

    public void closeConnection(){
        System.out.println("关闭连接!");
    }
}


二. 上传方法服务

我们假设读取文件瞬间完成,又因建立连接需要5秒时间,此时应该等待获取连接后依次上传文件,所以我们通过无限循环调用承诺者对象FtpPromise.isDone()判断是否完成连接,如果连接完成开始上传文件。此处我们可以通过次数或时间来限制最大等待时间

public class UploadToFTP {
        FutureTask<FTPConnectionUtil> FtpPromise = FTPConnectionUtil.getPromise();
        System.out.println("读取文件开始!");
        ArrayList<File> files = new ArrayList<>();
        files.add(new File("file1"));
        files.add(new File("file2"));
        files.add(new File("file3"));
        System.out.println("读取文件结束!");
        FTPConnectionUtil ftpConnectionUtil = null;
        while (true) {
            Thread.sleep(1000);
            if(FtpPromise.isDone()){
                ftpConnectionUtil = FtpPromise.get();
                if (ftpConnectionUtil.getException() != null) {
                    throw ftpConnectionUtil.getException();
                }

                for (File file : files) {
                    ftpConnectionUtil.upload(file);
                }
                break;
            }else{
                System.out.println("等待");
            }
        }
        ftpConnectionUtil.closeConnection();
    }
}

三. 测试方法

public class Run {

    public static void main(String[] args) throws Exception {
        UploadToFTP uploadToFTP = new UploadToFTP();
        uploadToFTP.upload();
    }
}

四. 测试结果

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

推荐阅读更多精彩内容

  • 一、Python简介和环境搭建以及pip的安装 4课时实验课主要内容 【Python简介】: Python 是一个...
    _小老虎_阅读 5,738评论 0 10
  • 1、第八章 Samba服务器2、第八章 NFS服务器3、第十章 Linux下DNS服务器配站点,域名解析概念命令:...
    哈熝少主阅读 3,729评论 0 10
  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,094评论 1 32
  • 这篇文章主要是对多线程的问题进行总结的,因此罗列了40个多线程的问题。 这些多线程的问题,有些来源于各大网站、有些...
    濡沫笔记阅读 924评论 0 1
  • 一、命令行 1. calc-----------启动计算器 2.certmgr.msc----证书管理实用程序 3...
    小小辛_c阅读 703评论 0 2