019.狀態(tài)模式

我們每天都在乘電梯,那我們來看看電梯有哪些動作(映射到 Java 中就是有多少方法):開門推沸、關(guān)門备绽、運(yùn)行、停止鬓催,就這四個動作肺素,好,我們就用程序來實(shí)現(xiàn)一下電梯的動作宇驾,先看類圖設(shè)計(jì):

代碼如下:

/**
 * @description 定義一個電梯的接口
 */
public interface ILift {

    /**
     * 電梯門開啟動作
     */
    void open();

    /**
     * 電梯門關(guān)閉動作
     */
    void close();

    /**
     * 電梯上升或下降
     */
    void run();

    /**
     * 電梯停在某層
     */
    void stop();

}

public class Lift implements ILift {

    @Override
    public void open() {
        System.out.println("電梯門打開...");
    }

    @Override
    public void close() {
        System.out.println("電梯門關(guān)閉...");
    }

    @Override
    public void run() {
        System.out.println("電梯上升或下降...");
    }

    @Override
    public void stop() {
        System.out.println("電梯停在某層...");
    }
}

public class Client {

    public static void main(String[] args) {

        ILift lift = new Lift();
        lift.open();
        lift.close();
        lift.run();
        lift.stop();

    }

}

這個程序有什么問題倍靡,你想呀電梯門可以打開,但不是隨時都可以開飞苇,是有前提條件的的菌瘫,你不可能電梯在運(yùn)行的時候突然開門吧?布卡!電梯也不會出現(xiàn)停止了但是不開門的情況吧雨让?!電梯的這四個動作的執(zhí)行都是有前置條件忿等,具體點(diǎn)說在特定狀態(tài)下才能做特定事栖忠,那我們來分析一下電梯有什么那些特定狀態(tài):

  • 門敞狀態(tài):按了電梯上下按鈕,電梯門開贸街,這中間有5秒的時間庵寞,在這個狀態(tài)下電梯只能做的動作是關(guān)門動作,做別的動作那就危險(xiǎn)
  • 門閉狀態(tài):電梯門關(guān)閉了薛匪,在這個狀態(tài)下捐川,可以進(jìn)行的動作是:開門(我不想坐電梯了)、停止(忘記按樓層號了)逸尖、運(yùn)行
  • 運(yùn)行狀態(tài):電梯正在跑古沥,上下竄,在這個狀態(tài)下娇跟,電梯只能做的是停止岩齿;
  • 停止?fàn)顟B(tài):電梯停止不動,在這個狀態(tài)下苞俘,電梯有兩個可選動作:繼續(xù)運(yùn)行或者開門盹沈;

我們用一張表來表示電梯狀態(tài)和動作之間的關(guān)系:


好,我們來修改一下吃谣,先看類圖:

在接口中定義了四個常量乞封,分別表示電梯的四個狀態(tài):門敞狀態(tài)做裙、關(guān)閉狀態(tài)、運(yùn)行狀態(tài)歌亲、停止?fàn)顟B(tài)菇用,然后在實(shí)現(xiàn)類中電梯的每一次動作發(fā)生都要對狀態(tài)進(jìn)行判斷,判斷是否運(yùn)行執(zhí)行陷揪,也就是動作的執(zhí)行是否符合業(yè)務(wù)邏輯惋鸥,實(shí)現(xiàn)類中的四個私有方法是僅僅實(shí)現(xiàn)電梯的動作,沒有任何的前置條件悍缠,因此這四個方法是不能為外部類調(diào)用的卦绣,設(shè)置為私有方法。代碼如下:

public interface ILift {

    /**
     * 門敞狀態(tài)
     */
    int OPENING_STATE = 1;

    /**
     * 門閉狀態(tài)
     */
    int CLOSING_STATE = 2;

    /**
     * 運(yùn)行狀態(tài)
     */
    int RUNNING_STATE = 3;

    /**
     * 設(shè)置電梯的狀態(tài)
     * @param state 電梯狀態(tài)
     */
    void setState(int state);

    /**
     * 停止?fàn)顟B(tài)
     */
    int STOPPING_STATE = 4;

    /**
     * 電梯門開啟動作
     */
    void open();

    /**
     * 電梯門關(guān)閉動作
     */
    void close();

    /**
     * 電梯上升或下降
     */
    void run();

    /**
     * 電梯停在某層
     */
    void stop();

}

public class Lift implements ILift {

    private int state;

    @Override
    public void setState(int state) {
        this.state = state;
    }

    @Override
    public void open() {
        // 在關(guān)門狀態(tài)和停止?fàn)顟B(tài)可以開門飞蚓,其他狀態(tài)什么也不做
        switch (state) {
            case CLOSING_STATE:
            case STOPPING_STATE:
                openWithoutLogic();
                setState(OPENING_STATE);
                break;
            case OPENING_STATE:
            case RUNNING_STATE:
                // do nothing
                break;
            default: // do nothing
        }
    }

    @Override
    public void close() {
        // 只有在開門狀態(tài)可以關(guān)門
        switch (state) {
            case OPENING_STATE:
                closeWithoutLogic();
                setState(CLOSING_STATE);
                break;
            case CLOSING_STATE:
            case STOPPING_STATE:
            case RUNNING_STATE:
                // do nothing
                break;
            default: // do nothing
        }
    }

    @Override
    public void run() {
        // 只有在關(guān)閉狀態(tài)和停止?fàn)顟B(tài)可以運(yùn)行
        switch (state) {
            case CLOSING_STATE:
            case STOPPING_STATE:
                runWithoutLogic();
                setState(RUNNING_STATE);
                break;
            case OPENING_STATE:
            case RUNNING_STATE:
                // do nothing
                break;
            default: // do nothing
        }
    }

    @Override
    public void stop() {
        // 只有在關(guān)閉狀態(tài)和運(yùn)行狀態(tài)可以停止
        switch (state) {
            case CLOSING_STATE:
            case RUNNING_STATE:
                stopWithoutLogic();
                setState(STOPPING_STATE);
                break;
            case OPENING_STATE:
            case STOPPING_STATE:
                // do nothing
                break;
            default: // do nothing
        }
    }

    /**
     * 單純的打開門滤港,不考慮其他條件
     */
    private void openWithoutLogic() {
        System.out.println("電梯門打開...");
    }

    /**
     * 單純的關(guān)閉門,不考慮其他條件
     */
    private void closeWithoutLogic() {
        System.out.println("電梯門關(guān)閉...");
    }

    /**
     * 單純的運(yùn)行趴拧,不考慮其他條件
     */
    private void runWithoutLogic() {
        System.out.println("電梯上升或下降...");
    }

    /**
     * 單純的停止溅漾,不考慮其他條件
     */
    private void stopWithoutLogic() {
        System.out.println("電梯停在某層...");
    }

}

public class Client {

    public static void main(String[] args) {

        ILift lift = new Lift();
        // 電梯的初始狀態(tài)為停止?fàn)顟B(tài)
        lift.setState(ILift.STOPPING_STATE);
        lift.open();
        lift.close();
        lift.run();
        lift.stop();

    }

}

以上的代碼也是有問題的:

  • 首先Lift這個類有點(diǎn)長,長的原因是我們在程序中使用了大量的switch…case這樣的判斷(if…else也是一樣)著榴,程序中只要你有這樣的判斷就避免不了加長程序添履,同步的在業(yè)務(wù)比較復(fù)雜的情況下,程序體會更長脑又,這個就不是一個很好的習(xí)慣了暮胧,較長的方法或者
    類的維護(hù)性比較差
  • 其次,擴(kuò)展性非常的不好问麸,大家來想想往衷,電梯還有兩個狀態(tài)沒有加,是什么严卖?通電狀態(tài)和斷電狀態(tài)席舍,你要是在程序再增加這兩個方法,你看看open()哮笆、close()俺亮、run()stop()這四個方法都要增加判斷條件疟呐,也就是說switch斷體中還要增加case項(xiàng),也就說與開閉原則相違背了东且;
  • 再其次启具,我們來思考我們的業(yè)務(wù),電梯在門敞開狀態(tài)下就不能上下跑了嗎珊泳?電梯有沒有發(fā)生過只有運(yùn)行沒有停止?fàn)顟B(tài)呢(從 40 層直接墜到 1 層嘛)鲁冯?電梯故障拷沸,還有電梯在檢修的時候,可以在stop狀態(tài)下不開門薯演,這也是正常的業(yè)務(wù)需求呀撞芍,你想想看,如果加上這些判斷條件跨扮,上面的程序有多少需要修改序无?看看我們的類,業(yè)務(wù)上的任務(wù)一個小小增加或改動都對我們的這個電梯類產(chǎn)生了修改衡创,這是在項(xiàng)目開發(fā)上是有很大風(fēng)險(xiǎn)的

我們來思考兩個問題:

  • 第一帝嗡、這個停止?fàn)顟B(tài)時怎么來的,那當(dāng)然是由于電梯執(zhí)行了stop()方法而來的璃氢;
  • 第二哟玷、在停止?fàn)顟B(tài)下,電梯還能做什么動作?繼續(xù)運(yùn)行一也?開門巢寡?那當(dāng)然都可以了。

我們再來分析其他三個狀態(tài)椰苟,也都是一樣的結(jié)果抑月,我們只要實(shí)現(xiàn)電梯在一個狀態(tài)下的兩個任務(wù)模型就可以了:這個狀態(tài)是如何產(chǎn)生的以及在這個狀態(tài)下還能做什么其他動作(也就是這個狀態(tài)怎么過渡到其他狀態(tài)),既然我們以狀態(tài)為參考模型尊剔,那我們就先定義電梯的狀態(tài)接口爪幻,思考過后我們來看類圖:

在類圖中,定義了一個LiftState抽象類须误,聲明了一個受保護(hù)的類型Context變量挨稿,這個是串聯(lián)我們各個狀態(tài)的封裝類,封裝的目的很明顯京痢,就是電梯對象內(nèi)部狀態(tài)的變化不被調(diào)用類知曉奶甘,也就是迪米特法則了,我的類內(nèi)部情節(jié)你知道越少越好祭椰,并且還定義了四個具體的實(shí)現(xiàn)類臭家,承擔(dān)的是狀態(tài)的產(chǎn)生以及狀態(tài)間的轉(zhuǎn)換過渡,全部的代碼如下:

public abstract class LiftState {

    /**
     * 定義一個環(huán)境角色方淤,也就是封裝狀態(tài)的變換引起的功能變化
     */
    protected Context context;

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

    /**
     * 電梯門開啟動作
     */
    public abstract void open();

    /**
     * 電梯門關(guān)閉動作
     */
    public abstract void close();

    /**
     * 電梯運(yùn)行動作
     */
    public abstract void run();

    /**
     * 電梯停止動作
     */
    public abstract void stop();

}

public class Context {

    /**
     * 定義出所有的電梯狀態(tài)
     */
    public static final OpeningState OPENING_STATE = new OpeningState();
    public static final ClosingState CLOSING_STATE = new ClosingState();
    public static final RunningState RUNNING_STATE = new RunningState();
    public static final StoppingState STOPPING_STATE = new StoppingState();

    /**
     * 定義一個當(dāng)前電梯狀態(tài)
     */
    private LiftState liftState;

    public LiftState getLiftState() {
        return liftState;
    }

    public void setLiftState(LiftState liftState) {
        this.liftState = liftState;
        // 把當(dāng)前環(huán)境通知到各個實(shí)現(xiàn)類中
        this.liftState.setContext(this);
    }

    public void open() {
        liftState.open();
    }

    public void close() {
        liftState.close();
    }

    public void run() {
        liftState.run();
    }

    public void stop() {
        liftState.stop();
    }

}

public class OpeningState extends LiftState {

    @Override
    public void open() {
        System.out.println("電梯門開啟...");
    }

    @Override
    public void close() {
        // 修改狀態(tài)
        super.context.setLiftState(Context.CLOSING_STATE);
        // 動作委托CLOSING_STATE來執(zhí)行
        super.context.getLiftState().close();
    }

    /**
     * 打開狀態(tài)不能運(yùn)行
     */
    @Override
    public void run() {
        // do nothing
    }

    /**
     * 打開狀態(tài)是停止?fàn)顟B(tài)
     */
    @Override
    public void stop() {
        // do nothing
    }
}

public class ClosingState extends LiftState {

    @Override
    public void open() {
        super.context.setLiftState(Context.OPENING_STATE);
        super.context.getLiftState().open();
    }

    @Override
    public void close() {
        System.out.println("電梯門關(guān)閉...");
    }

    @Override
    public void run() {
        super.context.setLiftState(Context.RUNNING_STATE);
        super.context.getLiftState().run();
    }

    @Override
    public void stop() {
        super.context.setLiftState(Context.STOPPING_STATE);
        super.context.getLiftState().stop();
    }
}

public class RunningState extends LiftState {

    @Override
    public void open() {
        // do nothing
    }

    @Override
    public void close() {
        // do nothing
    }

    @Override
    public void run() {
        System.out.println("電梯上下跑...");
    }

    @Override
    public void stop() {
        super.context.setLiftState(Context.STOPPING_STATE);
        super.context.getLiftState().stop();
    }
}

public class StoppingState extends LiftState {

    @Override
    public void open() {
        super.context.setLiftState(Context.OPENING_STATE);
        super.context.getLiftState().open();
    }

    @Override
    public void close() {
        // do nothing
    }

    @Override
    public void run() {
        super.context.setLiftState(Context.RUNNING_STATE);
        super.context.getLiftState().run();
    }

    @Override
    public void stop() {
        System.out.println("電梯停止了...");
    }
}

public class Client {

    public static void main(String[] args) {

        Context context = new Context();
        context.setLiftState(new ClosingState());
        context.open();
        context.close();
        context.run();
        context.stop();

    }

}

我們可以這樣理解钉赁,Context是一個環(huán)境角色,它的作用是串聯(lián)各個狀態(tài)的過渡携茂,在LiftSate抽象類中我們定義了并把這個環(huán)境角色聚合進(jìn)來你踩,并傳遞到了子類,也就是四個具體的實(shí)現(xiàn)類中自己根據(jù)環(huán)境來決定如何進(jìn)行狀態(tài)的過渡。

我們再來回顧一下我們剛剛批判上一段的代碼带膜,首先我們說人家代碼太長吩谦,這個問題我們解決了,通過各個子類來實(shí)現(xiàn)膝藕,每個子類的代碼都很短式廷,而且也取消了的switch…case條件的判斷;

其次芭挽,說人家不符合開閉原則滑废,那如果在我們這個例子中要增加兩個狀態(tài)怎么加?增加兩個子類览绿,一個是通電狀態(tài)郊愧,一個是斷電狀態(tài)柄冲,同時修改其他實(shí)現(xiàn)類的相應(yīng)方法计技,因?yàn)闋顟B(tài)要過渡呀涌攻,那當(dāng)然要修改原有的類,只是在原有類中的方法上增加怀各,而不去做修改倔韭;

再其次,我們說人家不符合迪米特法則瓢对,我們現(xiàn)在呢是各個狀態(tài)是單獨(dú)的一個類寿酌,只有與這個狀態(tài)的有關(guān)的因素修改了這個類才修改,符合迪米特法則硕蛹,非常完美!

上面例子中多次提到狀態(tài)醇疼,那我們這節(jié)講的就是狀態(tài)模式,什么是狀態(tài)模式呢法焰?當(dāng)一個對象內(nèi)在狀態(tài)改變時允許其改變行為秧荆,這個對象看起來像是改變了其類。說實(shí)話埃仪,這個定義的后半句我也沒看懂乙濒,看過GOF才明白是怎么回事: Allow an object to alter its behavior when its internal state changes. The object will appear to change its class,也就是說狀態(tài)模式封裝的非常好卵蛉,狀態(tài)的變更引起了行為的變更颁股,從外部看起來就好像這個對象對應(yīng)的類發(fā)生了改變一樣。

狀態(tài)模式的通用實(shí)現(xiàn)類如下:

狀態(tài)模式中有什么優(yōu)點(diǎn)呢傻丝?

  • 首先是避免了過多的swith…case或者if...else語句的使用甘有,避免了程序的復(fù)雜性;
  • 其次是很好的使用體現(xiàn)了開閉原則和單一職責(zé)原則葡缰,每個狀態(tài)都是一個子類亏掀,你要增加狀態(tài)就增加子類允睹,你要修改狀態(tài),你只修改一個子類就可以了幌氮;
  • 最后一個好處就是封裝性非常好,這也是狀態(tài)模式的基本要求胁澳,狀態(tài)變換放置到了類的內(nèi)部來實(shí)現(xiàn)该互,外部的調(diào)用不用知道類內(nèi)部如何實(shí)現(xiàn)狀態(tài)和行為的變換。

狀態(tài)模式的缺點(diǎn):類會太多韭畸,也就是類膨脹宇智,一個事物有七八十來個狀態(tài)也不稀奇,如果完全使用狀態(tài)模式就會有太多的子類胰丁,不好管理随橘。

狀態(tài)模式使用于當(dāng)某個對象在它的狀態(tài)發(fā)生改變時,它的行為也隨著發(fā)生比較大的變化锦庸,也就是說行為是受狀態(tài)約束的情況下可以使用狀態(tài)模式机蔗,而且狀態(tài)模式使用時對象的狀態(tài)最好不要超過五個。

本文原書:

《您的設(shè)計(jì)模式》 作者:CBF4LIFE

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末甘萧,一起剝皮案震驚了整個濱河市萝嘁,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌扬卷,老刑警劉巖牙言,帶你破解...
    沈念sama閱讀 218,941評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異怪得,居然都是意外死亡咱枉,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,397評論 3 395
  • 文/潘曉璐 我一進(jìn)店門徒恋,熙熙樓的掌柜王于貴愁眉苦臉地迎上來蚕断,“玉大人,你說我怎么就攤上這事因谎』ǎ” “怎么了?”我有些...
    開封第一講書人閱讀 165,345評論 0 356
  • 文/不壞的土叔 我叫張陵财岔,是天一觀的道長风皿。 經(jīng)常有香客問我,道長匠璧,這世上最難降的妖魔是什么桐款? 我笑而不...
    開封第一講書人閱讀 58,851評論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮夷恍,結(jié)果婚禮上魔眨,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好遏暴,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,868評論 6 392
  • 文/花漫 我一把揭開白布侄刽。 她就那樣靜靜地躺著,像睡著了一般朋凉。 火紅的嫁衣襯著肌膚如雪州丹。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,688評論 1 305
  • 那天杂彭,我揣著相機(jī)與錄音墓毒,去河邊找鬼。 笑死亲怠,一個胖子當(dāng)著我的面吹牛所计,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播团秽,決...
    沈念sama閱讀 40,414評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼主胧,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了徙垫?” 一聲冷哼從身側(cè)響起讥裤,我...
    開封第一講書人閱讀 39,319評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎姻报,沒想到半個月后己英,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,775評論 1 315
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡吴旋,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,945評論 3 336
  • 正文 我和宋清朗相戀三年损肛,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片荣瑟。...
    茶點(diǎn)故事閱讀 40,096評論 1 350
  • 序言:一個原本活蹦亂跳的男人離奇死亡治拿,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出笆焰,到底是詐尸還是另有隱情劫谅,我是刑警寧澤,帶...
    沈念sama閱讀 35,789評論 5 346
  • 正文 年R本政府宣布嚷掠,位于F島的核電站捏检,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏不皆。R本人自食惡果不足惜贯城,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,437評論 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望霹娄。 院中可真熱鬧能犯,春花似錦鲫骗、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,993評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至渡蜻,卻和暖如春坦胶,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背晴楔。 一陣腳步聲響...
    開封第一講書人閱讀 33,107評論 1 271
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留峭咒,地道東北人税弃。 一個月前我還...
    沈念sama閱讀 48,308評論 3 372
  • 正文 我出身青樓,卻偏偏與公主長得像凑队,于是被迫代替她去往敵國和親则果。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,037評論 2 355

推薦閱讀更多精彩內(nèi)容