欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java設(shè)計模式之狀態(tài)模式詳解

 更新時間:2023年05月12日 10:38:12   作者:蜀山劍客李沐白  
Java?中的狀態(tài)模式(State?Pattern)是一種行為型設(shè)計模式,它允許對象在內(nèi)部狀態(tài)發(fā)生改變時改變其行為,本文將詳細(xì)介紹?Java?中的狀態(tài)模式,我們將從狀態(tài)模式的概述、結(jié)構(gòu)與實現(xiàn)、優(yōu)缺點、適用場景等方面進(jìn)行講解,需要的朋友可以參考下

1. 狀態(tài)模式的概述

狀態(tài)模式是一種通過將對象的狀態(tài)轉(zhuǎn)換邏輯分布到狀態(tài)對象中來實現(xiàn)狀態(tài)轉(zhuǎn)換的設(shè)計模式。它將對象的行為與對應(yīng)的狀態(tài)分離,使得在修改對象狀態(tài)時,不需要修改對象的行為方法。同時,狀態(tài)模式可以通過將狀態(tài)的轉(zhuǎn)換邏輯包含在各個狀態(tài)類中來簡化代碼,避免出現(xiàn)大量的條件判斷語句,從而提高代碼的可讀性和可維護(hù)性。

根據(jù) GoF 的定義,狀態(tài)模式的三個核心角色分別是:

  • 環(huán)境(Context):它定義了客戶端所感興趣的接口,并維護(hù)一個當(dāng)前狀態(tài),在具體狀態(tài)類中實現(xiàn)該接口的各個具體操作。
  • 抽象狀態(tài)(State):它定義了一個接口,用于封裝環(huán)境對象中不同狀態(tài)對應(yīng)的行為。
  • 具體狀態(tài)(Concrete State):它實現(xiàn)了抽象狀態(tài)接口,封裝了不同狀態(tài)下對環(huán)境對象的響應(yīng)行為。

2. 狀態(tài)模式的結(jié)構(gòu)與實現(xiàn)

在 Java 中,狀態(tài)模式的實現(xiàn)方法比較簡單,通??梢园凑找韵虏襟E進(jìn)行:

  • 定義抽象狀態(tài)接口(State),它包含了具體狀態(tài)所對應(yīng)的操作方法;
  • 定義具體狀態(tài)類(ConcreteState1、ConcreteState2等),它們實現(xiàn)了抽象狀態(tài)接口,封裝了具體的狀態(tài)行為;
  • 定義環(huán)境類(Context),它包含了當(dāng)前狀態(tài)以及對外提供的操作接口;
  • 在環(huán)境類中,使用一個State類型的變量來表示當(dāng)前狀態(tài),并在具體操作中調(diào)用該狀態(tài)的方法;
  • 當(dāng)狀態(tài)發(fā)生改變時,修改環(huán)境對象的狀態(tài)即可。

下面是 Java 中狀態(tài)模式的一個簡單實現(xiàn):

// 定義抽象狀態(tài)接口
interface State {
    void handle();
}

// 定義具體狀態(tài)類
class ConcreteState1 implements State {
    @Override
    public void handle() {
        System.out.println("當(dāng)前狀態(tài)為 State1.");
    }
}

class ConcreteState2 implements State {
    @Override
    public void handle() {
        System.out.println("當(dāng)前狀態(tài)為 State2.");
    }
}

// 定義環(huán)境類
class Context {
    private State state;

    public void setState(State state) {
        this.state = state;
    }

    public void request() {
        state.handle();
    }
}

// 示例程序
public class StatePatternDemo {
    public static void main(String[] args) {
        // 創(chuàng)建狀態(tài)對象
        State state1 = new ConcreteState1();
        State state2 = new ConcreteState2();

        // 創(chuàng)建環(huán)境對象
        Context context = new Context();
        context.setState(state1);
        context.request();

        context.setState(state2);
        context.request();
    }
}

在上述代碼中,我們首先定義了抽象狀態(tài)接口State和兩個具體狀態(tài)類ConcreteState1、ConcreteState2,它們分別實現(xiàn)了State接口。然后,我們定義了一個包含狀態(tài)切換邏輯的環(huán)境類Context,其中,使用狀態(tài)對象來表示當(dāng)前的狀態(tài),并在request方法中調(diào)用當(dāng)前狀態(tài)的handle方法。最后,我們創(chuàng)建一個示例程序,調(diào)用context的setState方法來改變狀態(tài),并觀察其輸出。

3. 狀態(tài)模式的優(yōu)缺點

狀態(tài)模式具有如下優(yōu)點:

  • 結(jié)構(gòu)清晰、封裝性好:將狀態(tài)的轉(zhuǎn)換邏輯分布到獨立的狀態(tài)類中,使得狀態(tài)之間的耦合度降低,并且可以將狀態(tài)的行為封裝在狀態(tài)類中,提高了系統(tǒng)的可維護(hù)性和可讀性。
  • 擴(kuò)展性好:對于新的狀態(tài),只需要創(chuàng)建一個具體狀態(tài)類即可,同時也可以很方便地增加新的狀態(tài)轉(zhuǎn)換。
  • 易于維護(hù)和調(diào)試:狀態(tài)模式將各個狀態(tài)進(jìn)行了封裝,每個狀態(tài)對象都只關(guān)注自身的行為,使得代碼易于維護(hù)和調(diào)試。

但是狀態(tài)模式也存在一些缺點:

  • 狀態(tài)模式會導(dǎo)致系統(tǒng)類和對象的個數(shù)增加:狀態(tài)模式將每個狀態(tài)都封裝成了獨立的對象,因此會增加系統(tǒng)的復(fù)雜度和實現(xiàn)難度。
  • 狀態(tài)模式的使用條件較為苛刻:由于狀態(tài)模式要求將狀態(tài)轉(zhuǎn)換邏輯包含在具體狀態(tài)類中,因此只適合“狀態(tài)不多”且“狀態(tài)轉(zhuǎn)換比較少”的情況,否則會導(dǎo)致系統(tǒng)的維護(hù)和擴(kuò)展變得困難。

4. 狀態(tài)模式的適用場景

狀態(tài)模式通常適用于以下情形:

  • 行為隨狀態(tài)改變而改變的場景:在狀態(tài)模式中,行為是由狀態(tài)決定的,因此當(dāng)一個對象狀態(tài)改變時,它所對應(yīng)的行為也隨之改變。
  • 條件、分支語句多的場景:如果使用傳統(tǒng)的if/else語句實現(xiàn)狀態(tài)轉(zhuǎn)換邏輯,通常會出現(xiàn)大量的條件語句,從而導(dǎo)致代碼復(fù)雜度的提高,而狀態(tài)模式可以很好地解決這個問題。

具體來說,狀態(tài)模式通常適用于以下場景:

  • 對象的行為取決于它的狀態(tài),并且它必須在運行時刻根據(jù)狀態(tài)改變它的行為;
  • 某個操作有多個狀態(tài),且這些狀態(tài)之間可以相互轉(zhuǎn)換;
  • 在不同狀態(tài)下執(zhí)行的操作有大量重復(fù)代碼時,可以將該重復(fù)代碼封裝在具體狀態(tài)類中,從而提高代碼的重用性和可維護(hù)性。

5. 示例程序的設(shè)計與實現(xiàn)

下面我們將使用一個簡單示例來說明狀態(tài)模式的具體實現(xiàn)方法。假設(shè)我們正在開發(fā)一個多線程下載器程序,該程序可以同時下載多個文件,并且可以監(jiān)控每個文件的下載進(jìn)度,當(dāng)某個文件下載完成后,該程序需要自動關(guān)閉該文件的下載線程并向用戶發(fā)送下載完成的提示信息。

為了實現(xiàn)上述功能,我們可以使用狀態(tài)模式對下載器程序進(jìn)行重構(gòu),具體設(shè)計如下:

  • 定義抽象狀態(tài)類,并聲明抽象的 download 方法,用于封裝不同狀態(tài)下的共性操作。
  • 定義具體狀態(tài)類,并實現(xiàn) download 方法,用于完成具體的狀態(tài)操作邏輯。
  • 在 ConcreteState 類中,定義一個靜態(tài)變量來表示當(dāng)前狀態(tài),在 download 方法中根據(jù)下載狀態(tài)進(jìn)行狀態(tài)轉(zhuǎn)換。
  • 在 Context 類中,維護(hù)一個當(dāng)前狀態(tài),并將 download 方法委托給當(dāng)前狀態(tài)對象來執(zhí)行。

接下來我們來看一下示例程序的具體實現(xiàn)。在本示例程序中,我們使用了 Java 中的線程池和 FutureTask,實現(xiàn)了對多個文件的同時下載。需要注意的是,由于本文篇幅較長,為了讓代碼更加清晰,我們將代碼拆分成了多個類來實現(xiàn)相應(yīng)的功能。

(1)抽象狀態(tài)類

public abstract class DownloadState {
    protected DownloadContext context;

    public void setContext(DownloadContext context) {
        this.context = context;
    }

    public abstract void download(String url, String filePath);
}

在上述代碼中,我們定義了一個抽象狀態(tài)類 DownloadState,它包含了一個 DownloadContext 對象,以及一個 download 方法,用于封裝不同狀態(tài)下的下載操作。需要注意的是,該抽象方法不包含具體的下載邏輯,具體的下載邏輯需要在具體狀態(tài)類中進(jìn)行實現(xiàn)。

(2)具體狀態(tài)類

public class DownloadingState extends DownloadState {
    private FutureTask<Integer> futureTask;

    @Override
    public void download(String url, String filePath) {
        System.out.println("開始下載文件:" + filePath);

        // 開始下載
        DownloadTask task = new DownloadTask(url, filePath);
        futureTask = new FutureTask<>(task);
        ThreadPool.getInstance().execute(futureTask);

        // 狀態(tài)轉(zhuǎn)換
        try {
            int result = futureTask.get();
            if (result == 0) {
                context.setState(new FinishedState());
            } else {
                context.setState(new ErrorState());
            }
        } catch (Exception e) {
            e.printStackTrace();
            context.setState(new ErrorState());
        }
    }
}

public class FinishedState extends DownloadState {
    @Override
    public void download(String url, String filePath) {
        System.out.println("文件已下載完成,無需重復(fù)下載!");
        context.closeDownloadThread(filePath);
    }
}

public class ErrorState extends DownloadState {
    @Override
    public void download(String url, String filePath) {
        System.out.println("下載文件出錯,無法繼續(xù)下載!");
        context.closeDownloadThread(filePath);
    }
}

在上述代碼中,我們定義了三個具體狀態(tài)類:DownloadingState、FinishedState 和 ErrorState,它們分別代表下載中、下載完成和下載出錯三種狀態(tài)。其中,我們使用了線程池和 FutureTask 來實現(xiàn)下載操作,并根據(jù)下載結(jié)果進(jìn)行狀態(tài)轉(zhuǎn)換(對于下載成功的情況,我們轉(zhuǎn)換到FinishedState;對于下載出錯的情況,我們轉(zhuǎn)換到ErrorState)。

需要注意的是,由于下載完成或下載出錯后都需要關(guān)閉下載線程,因此我們在FinishedState和ErrorState中都調(diào)用了context對象的closeDownloadThread方法來實現(xiàn)該功能。

(3)環(huán)境類

public class DownloadContext {
    private DownloadState currentState;
    private Map<String, FutureTask<Integer>> taskMap;

    public DownloadContext() {
        this.currentState = new FinishedState();
        this.taskMap = new HashMap<>();
      }

      public void setState(DownloadState state) {
        this.currentState = state;
        this.currentState.setContext(this);
      }

      public void download(String url, String filePath) {
        FutureTask<Integer> task = this.taskMap.get(filePath);

        if (task == null || task.isDone() || task.isCancelled()) {
          this.taskMap.remove(filePath);
          this.currentState.download(url, filePath);
        } else {
          System.out.println("文件 " + filePath + " 正在下載中,無需重復(fù)下載!");
        }
      }

      public void closeDownloadThread(String filePath) {
        FutureTask<Integer> task = this.taskMap.get(filePath);

        if (task != null) {
          task.cancel(true);
          this.taskMap.remove(filePath);
          System.out.println("已關(guān)閉文件 " + filePath + " 的下載線程。");
        }
      }
}

在上述代碼中,我們定義了一個 DownloadContext 類,它包含了當(dāng)前狀態(tài)以及 download 和 closeDownloadThread 方法。

在 download 方法中,我們首先檢查是否存在正在下載的任務(wù)(即 task 對象是否存在且未完成),如果不存在,則將當(dāng)前狀態(tài)轉(zhuǎn)換為下載中狀態(tài),并啟動下載任務(wù);否則輸出提示信息,防止重復(fù)下載。

在 closeDownloadThread 方法中,我們將傳入的 filePath 對應(yīng)的下載任務(wù)取消,并從 taskMap 中移除該任務(wù),同時輸出提示信息。

(4)線程池類

public class ThreadPool {
    private ExecutorService executor;

    private ThreadPool() {
        this.executor = Executors.newFixedThreadPool(5);
    }

    private static class Singleton {
        private static final ThreadPool INSTANCE = new ThreadPool();
    }

    public static ThreadPool getInstance() {
        return Singleton.INSTANCE;
    }

    public void execute(Runnable task) {
        this.executor.execute(task);
    }
}

在上述代碼中,我們定義了一個 ThreadPool 類,它包含了一個靜態(tài)的 ExecutorService 對象 executor,并封裝了一個 execute 方法用于提交任務(wù)到線程池中。

需要注意的是,由于我們要將ThreadPool類設(shè)計為單例模式,因此我們在該類中定義了一個私有的靜態(tài)內(nèi)部類Singleton,用于實現(xiàn)懶漢式單例模式。這樣可以保證線程池中只有一個實例對象,并且線程安全。

(5)示例程序

public class Main {
    public static void main(String[] args) {
        DownloadContext context = new DownloadContext();

        String url1 = "https://cdn.pixabay.com/photo/2018/10/30/16/06/water-lily-3784022__340.jpg";
        String filePath1 = "water-lily.jpg";

        String url2 = "https://cdn.pixabay.com/photo/2020/07/14/13/10/excursion-5407227__340.jpg";
        String filePath2 = "excursion.jpg";

        context.download(url1, filePath1);
        context.download(url2, filePath2);

        System.out.println("------------------------------------");

        context.download(url1, filePath1);
        context.download(url2, filePath2);
    }
}

在上述代碼中,我們創(chuàng)建了一個 DownloadContext 對象,并分別下載了兩個文件。需要注意的是,在第二次下載同一個文件時,系統(tǒng)會輸出提示信息“文件正在下載中,無需重復(fù)下載!”。

運行該程序,我們可以看到如下輸出結(jié)果:

開始下載文件:water-lily.jpg
開始下載文件:excursion.jpg
------------------------------------
文件正在下載中,無需重復(fù)下載!
文件正在下載中,無需重復(fù)下載!

從輸出結(jié)果中我們可以看出,根據(jù)不同的狀態(tài),下載器程序完成了不同的操作,并且順利地將多線程下載操作與狀態(tài)轉(zhuǎn)換功能封裝在了不同的狀態(tài)類中。

以上就是Java設(shè)計模式之狀態(tài)模式詳解的詳細(xì)內(nèi)容,更多關(guān)于Java狀態(tài)模式的資料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • Java利用正則表達(dá)式提取數(shù)據(jù)的方法

    Java利用正則表達(dá)式提取數(shù)據(jù)的方法

    最近由于項目需求需要提取txt里的數(shù)據(jù),之前用C#實現(xiàn)過,由于最近學(xué)習(xí)了java,所以嘗試用java實現(xiàn)下,這篇文章主要介紹了Java利用正則表達(dá)式提取數(shù)據(jù)的方法,需要的朋友可以參考下,下面來一起看看吧。
    2016-12-12
  • Java Spring快速入門

    Java Spring快速入門

    本文主要介紹了SpringSpring簡介和入門知識。具有一定的參考價值,下面跟著小編一起來看下吧
    2017-01-01
  • spring cloud gateway 如何修改請求路徑Path

    spring cloud gateway 如何修改請求路徑Path

    這篇文章主要介紹了spring cloud gateway 修改請求路徑Path的操作,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-06-06
  • java線程阻塞中斷與LockSupport使用介紹

    java線程阻塞中斷與LockSupport使用介紹

    本文將詳細(xì)介紹java線程阻塞中斷和LockSupport的使用,需要了解更多的朋友可以參考下
    2012-12-12
  • SpringBoot2.x的依賴管理配置

    SpringBoot2.x的依賴管理配置

    這篇文章主要介紹了SpringBoot2.x的依賴管理配置,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-06-06
  • logback的isDebugEnabled日志配置級別源碼解析

    logback的isDebugEnabled日志配置級別源碼解析

    這篇文章主要為大家介紹了logback的isDebugEnabled日志配置級別源碼解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-11-11
  • 解析Java中的static關(guān)鍵字

    解析Java中的static關(guān)鍵字

    static是方便在沒有創(chuàng)建對象的情況下進(jìn)行調(diào)用(方法/變量)。顯然,被static關(guān)鍵字修飾的方法或者變量不需要依賴于對象來進(jìn)行訪問,只要類被加載了,就可以通過類名去進(jìn)行訪問。static可以用來修飾類的成員方法、類的成員變量,另外也可以編寫static代碼塊來優(yōu)化程序性能
    2021-06-06
  • springboot 實現(xiàn)mqtt物聯(lián)網(wǎng)的示例代碼

    springboot 實現(xiàn)mqtt物聯(lián)網(wǎng)的示例代碼

    這篇文章主要介紹了springboot 實現(xiàn)mqtt物聯(lián)網(wǎng),本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2021-03-03
  • Java中對話框的彈出方法

    Java中對話框的彈出方法

    下面小編就為大家?guī)硪黄狫ava中對話框的彈出方法。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2016-10-10
  • Java在Word中插入上標(biāo)和下標(biāo)的實現(xiàn)方法

    Java在Word中插入上標(biāo)和下標(biāo)的實現(xiàn)方法

    在某些情況下,你可能需要在Microsoft?Word中插入上標(biāo)和下標(biāo)。例如,當(dāng)你正在創(chuàng)建一個涉及科學(xué)公式的學(xué)術(shù)文件時,在這篇文章中,你將學(xué)習(xí)如何使用Spire.Doc?for?Java庫在Word文檔中插入上標(biāo)和下標(biāo),需要的朋友可以參考下
    2022-10-10

最新評論