怎麼用wait、notify巧妙的設計一個Future模式?

来源:https://www.cnblogs.com/starry-skys/archive/2020/02/23/12354336.html
-Advertisement-
Play Games

我們知道多線程可以實現同時執行多個任務(只是看起來是同時,其實是CPU的時間片切換特別快我們沒感覺而已)。 現在假設一個做飯的場景,你沒有廚具也沒有食材。你可以去網上買一個廚具,但是這段時間,你不需要閑著啊,可以同時去超市買食材。 設想這是兩個線程,主線程去買食材,然後開啟一個子線程去買廚具。但是, ...


我們知道多線程可以實現同時執行多個任務(只是看起來是同時,其實是CPU的時間片切換特別快我們沒感覺而已)。

現在假設一個做飯的場景,你沒有廚具也沒有食材。你可以去網上買一個廚具,但是這段時間,你不需要閑著啊,可以同時去超市買食材。

設想這是兩個線程,主線程去買食材,然後開啟一個子線程去買廚具。但是,子線程是需要返回一個廚具的。 如果用普通的線程,只有一個Run方法,而Run方法是沒有返回值的,這個時候該怎麼辦呢?

我們就可以用JDK提供的Future模式。在主線程買完食材之後,可以主動去獲取子線程的廚具。(本文認為讀者瞭解Future,因此不對Future用法做過多介紹)

代碼如下:

public class FutureCook {
    static class Chuju {

    }

    static class Shicai{

    }

    public static void cook(Chuju chuju,Shicai shicai){
        System.out.println("最後:烹飪中...");
    }

    public static void main(String[] args) throws InterruptedException, ExecutionException {
        //第一步,網購廚具
        Callable<Chuju> shopping = new Callable<Chuju>(){

            @Override
            public Chuju call() throws Exception {
                System.out.println("第一步:下單");
                System.out.println("第一步:等待送貨");
                Thread.sleep(5000); //模擬送貨時間
                System.out.println("第一步:快遞送到");
                return new Chuju();
            }
        };

        FutureTask<Chuju> task = new FutureTask<Chuju>(shopping);
        new Thread(task).start();

        //第二步,購買食材
        Thread.sleep(2000);
        Shicai shicai = new Shicai();
        System.out.println("第二步:食材到位");

        //第三步,烹飪
        if(!task.isDone()){ //是否廚具到位
            System.out.println("第三步:廚具還沒到,請等待,也可以取消");
                        //①
//            task.cancel(true);
//            System.out.println("已取消");
//            return;
        }

        //嘗試獲取結果,如果獲取不到,就會進入等待狀態
        // 即main線程等待子線程執行結束才能繼續往下執行
        Chuju chuju = task.get();
        System.out.println("第三步:廚具到位,可以烹飪了");

        cook(chuju,shicai);

    }
}

返回結果:

第一步:下單
第一步:等待送貨
第二步:食材到位
第三步:廚具還沒到,請等待,也可以取消
第一步:快遞送到
第三步:廚具到位,可以烹飪了
最後:烹飪中...

以上代碼表示,子線程購買廚具消耗的時間比較長(假定5秒),而主線程購買食材比較快(2秒),所以我在第三步烹飪之前,先去判斷一下買廚具的線程是否執行完畢。此處肯定返回false,然後主線程可以選擇繼續等待,也可以選擇取消。(把①註釋打開即可測試取消)

我們可以看到,利用Future模式,可以把原本同步執行的任務改為非同步執行,可以充分利用CPU資源,提高效率。

現在,我用wait、notify的方式來實現和以上Future模式一模一樣的效果。

大概思想就是,創建一個FutureClient端去發起請求,通過FutureData先立即返回一個結果(此時相當於只返回一個請求成功的通知),然後再去開啟一個線程非同步地執行任務,獲取真實數據RealData。此時,主線程可以繼續執行其他任務,當需要數據的時候,就可以調用get方法拿到真實數據。

1)定義一個數據介面,包含獲取數據的get方法,判斷任務是否執行完畢的isDone方法,和取消任務的cancel方法。

public interface Data<T> {

    T get();

    boolean isDone();

    boolean cancel();
}

2)定義真實數據的類,實現Data介面,用來執行實際的任務和返回真實數據。

public class RealData<T> implements Data<T>{

    private T result ;
    
    public RealData (){
        this.prepare();
    }

    private void prepare() {
        //準備數據階段,只有準備完成之後才可以繼續往下走
        try {
            System.out.println("第一步:下單");
            System.out.println("第一步:等待送貨");
            Thread.sleep(5000);
            System.out.println("第一步:快遞送到");
        } catch (InterruptedException e) {
            System.out.println("被中斷:"+e);
            //重新設置中斷狀態
            Thread.currentThread().interrupt();
        }
        Main.Chuju chuju = new Main.Chuju();
        result = (T)chuju;
    }

    @Override
    public T get() {
        return result;
    }

    @Override
    public boolean isDone() {
        return false;
    }

    @Override
    public boolean cancel() {
        return true;
    }

}

prepare方法用來準備數據,其實就是執行的實際任務。get方法用來返回任務的執行結果。

3)定義一個代理類FutureData用於給請求端FutureClient暫時返回一個假數據。等真實數據拿到之後,再裝載真實數據。

public class FutureData<T> implements Data<T>{

    private RealData<T> realData ;
    
    private boolean isReady = false;

    private Thread runningThread;
    
    public synchronized void setRealData(RealData realData) {
        //如果已經裝載完畢了,就直接返回
        if(isReady){
            return;
        }
        //如果沒裝載,進行裝載真實對象
        this.realData = realData;
        isReady = true;
        //進行通知
        notify();
    }
    
    @Override
    public synchronized T get() {
        //如果沒裝載好 程式就一直處於阻塞狀態
        while(!isReady){
            try {
                wait();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
        //裝載好直接獲取數據即可
        return realData.get();
    }


    public boolean isDone() {
        return isReady;
    }

    @Override
    public boolean cancel() {
        if(isReady){
            return false;
        }
        runningThread.interrupt();
        return true;
    }

    public void setRunningThread(){
        runningThread = Thread.currentThread();
    }
}

如果get方法被調用,就會去判斷數據是否已經被載入好(即判斷isReady的值),如果沒有的話就調用wait方法進入等待。

setRealData用於去載入真實的數據,載入完畢之後就把isReady設置為true,然後調用notify方法通知正在等待的線程。此時,get方法收到通知就繼續執行,然後返回真實數據realData.get().

另外也簡單的實現了一個取消任務的方法cancel,去中斷正在執行子任務的線程。

4)FutureClient客戶端用於發起請求,非同步執行任務。

public class FutureClient {

    public Data call(){
        //創建一個代理對象FutureData,先返回給客戶端(無論是否有值)
        final FutureData futureData = new FutureData();
        //啟動一個新的線程,去非同步載入真實的對象
        new Thread(new Runnable() {
            @Override
            public void run() {
                //此處註意需要記錄一下非同步載入真實數據的線程,以便後續可以取消任務。
                futureData.setRunningThread();
                RealData realData = new RealData();
                //等真實數據處理完畢之後,把結果賦值給代理對象
                futureData.setRealData(realData);
            }
        }).start();
        
        return futureData;
    }
    
}

5)測試

public class Main {

    static class Chuju{

    }

    static class Shicai{

    }

    public static void main(String[] args) throws InterruptedException {
        
        FutureClient fc = new FutureClient();
        Data data = fc.call();

        Thread.sleep(2000);
        Shicai shicai = new Shicai();
        System.out.println("第二步:食材到位");

        if(!data.isDone()){
            System.out.println("第三步:廚具還沒到,請等待或者取消");
            //②
//          data.cancel();
//            System.out.println("已取消");
//          return;
        }

        //真正需要數據的時候,再去獲取
        Chuju chuju = (Chuju)data.get();
        System.out.println("第三步:廚具到位,可以烹飪了");

        cook(chuju,shicai);

    }

    public static void cook (Chuju chuju, Shicai shicai){
        System.out.println("最後:烹飪中...");
    }
}

執行結果和用JDK提供的Future模式是一模一樣的。我們也可以把②出的代碼打開,測試任務取消的結果。

第一步:下單
第一步:等待送貨
第二步:食材到位
第三步:廚具還沒到,請等待或者取消
已取消
被中斷:java.lang.InterruptedException: sleep interrupted

執行取消之後,執行RealData的子線程就會被中斷,然後結束任務。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • mediainfo命令介紹 mediainfo.exe(Linux/iMac下是未帶尾碼的mediainfo), 是一款音視頻圖片文件的信息查詢工具, 常用於查看多媒體文件的視頻流信息,音頻流信息,字幕流信息等。 也可以用於查看圖片格式。 Windows版下載地址 "https://mediaare ...
  • 簡介: 正常情況下,HttpSession是通過Servlet 容器創建併進行管理的,創建成功之後都是保存在記憶體中。如果開發者需要對項目進行橫向擴展搭建集群,那麼可以利用一些硬體或者軟體工具來做負載均衡,此時,來自同一用戶的HTTP請求就有可能被分發到不同的實例上去,如何保證各個實例之間Sessio ...
  • ffmpeg命令介紹 ffmpeg.exe(linux/imac一般不帶尾碼,ffmpeg), 是一款音視頻編解碼的命令行工具軟體, 常用於多媒體測試的文件製作與轉碼。 我們常用的:格式工廠,MediaCoder等多媒體格式轉換軟體, 其關鍵技術有一部分是直接或間接依賴ffmpeg.exe的, ff ...
  • Java實例初始化程式 是在執行構造函數代碼之前執行的代碼塊。每當我們創建一個新對象時,這些初始化程式就會運行。 1.實例初始化語法 用 花括弧 創建實例初始化程式塊。對象初始化語句寫在括弧內。 2.Java實例初始化器功能 實例初始化器具有以下功能。 我們可以在一個類中定義 多個初始化器 。 所有 ...
  • Java構造函數 是特殊的方法(沒有返回類型),使您可以在應用程式內部的其他類使用對象之前完全初始化對象狀態。Java中的構造方法是使用 關鍵字調用的。下麵讓我們更深入地瞭解構造函數。 1.什麼是java構造函數 構造函數是一種特殊的方法,類似於(沒有確切的方法)構造,它可以幫助程式員在對象可被應用 ...
  • Java訪問修飾符–public, protected, private and default Java提供了 四個訪問修飾符 來設置類,變數,方法和構造函數的訪問級別,即 public , private , protected 和 default 。這些訪問級別修飾符確定其他類是否可以使用特定 ...
  • 在本 Java OOPs 概念教程中,我們將學習四種主要的面向對象原則 抽象、封裝、繼承和多態性。它們也被稱為面向對象編程範式的四大支柱。 1. _ 抽象 _是在不考慮無關細節的情況下公開實體基本細節的過程,以降低用戶的複雜性。 1. _ 封裝 _是將數據和對數據的操作捆綁到一個實體中的過程。 1. ...
  • 一、拋磚引玉 既然Java中支持以多線程的方式來執行相應的任務,但為什麼在JDK1.5中又提供了線程池技術呢?這個問題大家自行腦補,多動腦,肯定沒壞處,哈哈哈。。。 說起Java中的線程池技術,在很多框架和非同步處理中間件中都有涉及,而且性能經受起了長久的考驗。可以這樣說,Java的線程池技術是Jav ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...