JavaStatePattern(狀態模式)

凌浩雨發表於2017-09-10

在狀態模式(State Pattern)中,類的行為是基於它的狀態改變的。這種型別的設計模式屬於行為型模式。
在狀態模式中,我們建立表示各種狀態的物件和一個行為隨著狀態物件改變而改變的 context 物件。

關鍵程式碼:通常命令模式的介面中只有一個方法。而狀態模式的介面中有一個或者多個方法。而且,狀態模式的實現類的方法,一般返回值,或者是改變例項變數的值。也就是說,狀態模式一般和物件的狀態有關。實現類的方法有不同的功能,覆蓋介面中的方法。狀態模式和命令模式一樣,也可以用於消除 if…else 等條件選擇語句。

優點: 1、封裝了轉換規則。 2、列舉可能的狀態,在列舉狀態之前需要確定狀態種類。 3、將所有與某個狀態有關的行為放到一個類中,並且可以方便地增加新的狀態,只需要改變物件狀態即可改變物件的行為。 4、允許狀態轉換邏輯與狀態物件合成一體,而不是某一個巨大的條件語句塊。 5、可以讓多個環境物件共享一個狀態物件,從而減少系統中物件的個數。
缺點: 1、狀態模式的使用必然會增加系統類和物件的個數。 2、狀態模式的結構與實現都較為複雜,如果使用不當將導致程式結構和程式碼的混亂。 3、狀態模式對”開閉原則”的支援並不太好,對於可以切換狀態的狀態模式,增加新的狀態類需要修改那些負責狀態轉換的原始碼,否則無法切換到新增狀態,而且修改某個狀態類的行為也需修改對應類的原始碼。

注意事項:在行為受狀態約束的時候使用狀態模式,而且狀態不超過 5 個。

  1. 建立一個介面。
/**
 * 1. 建立一個介面
 * @author mazaiting
 */
public interface State {
    /**設定上下文*/
    void doAction(Context context);
}
  1. 建立Context類
/**
 * 2. 建立 Context 類。
 * @author mazaiting
 */
public class Context {
    
    private State state;
    
    public Context() {
        state = null;
    }

    public State getState() {
        return state;
    }

    public void setState(State state) {
        this.state = state;
    }   
}
  1. 建立實現介面的實體類。
/**
 * 3. 建立實現介面的實體類。
 * @author mazaiting
 */
public class StartState implements State{

    public void doAction(Context context) {
        System.out.println("Player is in start state.");
        context.setState(this);
    }

    @Override
    public String toString() {
        return "Start State";
    }
}


/**
 * 3. 建立實現介面的實體類。
 * @author mazaiting
 */
public class StopState implements State{

    public void doAction(Context context) {
        System.out.println("Player is in stop state.");
        context.setState(this);
    }

    @Override
    public String toString() {
        return "Stop State";
    }
}
  1. 使用 Context 來檢視當狀態 State 改變時的行為變化。
/**
 * 4. 使用 Context 來檢視當狀態 State 改變時的行為變化。
 * @author mazaiting
 */
public class Client {

    public static void main(String[] args) {
        
        Context context = new Context();
        
        StartState startState = new StartState();
        startState.doAction(context);
        
        System.out.println(context.getState().toString());
        
        StopState stopState = new StopState();
        stopState.doAction(context);
        
        System.out.println(context.getState().toString());
    }
}
  1. 列印結果
Player is in start state.
Start State
Player is in stop state.
Stop State


相關文章