設計模式之狀態者模式

来源:http://www.cnblogs.com/snaildev/archive/2017/10/19/7692342.html
-Advertisement-
Play Games

什麼是狀態者模式? 每個對象都有其對應的狀態,而每個狀態又對應一些相應的行為,如果某個對象有多個狀態時,那麼就會對應很多的行為。那麼對這些狀態的判斷和根據狀態完成的行為,就會導致多重條件語句,並且如果添加一種新的狀態時,需要更改之前現有的代碼。這樣的設計顯然違背了開閉原則。狀態模式正是用來解決這樣的 ...


什麼是狀態者模式?

每個對象都有其對應的狀態,而每個狀態又對應一些相應的行為,如果某個對象有多個狀態時,那麼就會對應很多的行為。那麼對這些狀態的判斷和根據狀態完成的行為,就會導致多重條件語句,並且如果添加一種新的狀態時,需要更改之前現有的代碼。這樣的設計顯然違背了開閉原則。狀態模式正是用來解決這樣的問題的。狀態模式將每種狀態對應的行為抽象出來成為單獨新的對象,這樣狀態的變化不再依賴於對象內部的行為。

狀態者模式:當一個對象的內在狀態改變時允許改變其行為,這個對象看起來像是改變了其類。其主要解決的是當控制一個對象狀態轉換的條件表達式過於複雜時的情況。通過把狀態的判斷邏輯轉移到表示不同的一系列類當中,可以把複雜的邏輯判斷簡單化。

  • 上下文環境(Context):它定義了客戶程式需要的介面並維護一個具體狀態角色的實例,將與狀態相關的操作委托給當前的Concrete State對象來處理。
  • 抽象狀態(State):定義一個介面以封裝使用上下文環境的的一個特定狀態相關的行為。
  • 具體狀態(Concrete State):實現抽象狀態定義的介面。

設計思路及代碼實現

/// <summary>
/// Context類,維護一個ConcreteState子類的實例,這個實例定義當前的狀態。
/// </summary>
public class Context
{
    private State state;
    /// <summary>
    /// 定義Context的初始狀態
    /// </summary>
    /// <param name="state"></param>
    public Context(State state)
    {
        this.state = state;
    }

    /// <summary>
    /// 可讀寫的狀態屬性,用於讀取和設置新狀態
    /// </summary>
    public State State
    {
        get { return state; }
        set { state = value; }
    }

    /// <summary>
    /// 對請求做處理,並設置下一個狀態
    /// </summary>
    public void Request()
    {
        state.Handle(this);
    }
}

/// <summary>
/// 抽象狀態類,定義一個介面以封裝與Context的一個特定狀態相關的行為
/// </summary>
public abstract class State
{
    public abstract void Handle(Context context);
}

/// <summary>
/// 具體狀態類,每一個子類實現一個與Context的一個狀態相關的行為
/// </summary>
public class ConcreteStateA : State
{
    /// <summary>
    /// 設置ConcreteStateA的下一個狀態是ConcreteStateB
    /// </summary>
    /// <param name="context"></param>
    public override void Handle(Context context)
    {
        Console.WriteLine("當前狀態是 A.");
        context.State = new ConcreteStateB();
    }
}

/// <summary>
/// 具體狀態類,每一個子類實現一個與Context的一個狀態相關的行為
/// </summary>
public class ConcreteStateB : State
{
    /// <summary>
    /// 設置ConcreteStateB的下一個狀態是ConcreteSateA
    /// </summary>
    /// <param name="context"></param>
    public override void Handle(Context context)
    {
        Console.WriteLine("當前狀態是 B.");
        context.State = new ConcreteStateA();
    }
}

/// <summary>
/// 調用
/// </summary>
/// <param name="args"></param>
static void Main(string[] args)
{
    // 設置Context的初始狀態為ConcreteStateA
    Context context = new Context(new ConcreteStateA());

    // 不斷地進行請求,同時更改狀態
    for (int i = 0; i < 10; i++)
    {
        context.Request();
    }


    Console.ReadLine();
}

應用狀態者模式完善中介者模式

/// <summary>
/// 抽象牌友類
/// </summary>
public abstract class AbstractCardPartner
{
    public int Money { get; set; }

    public abstract void ChangeMoney(int money, AbstractMediator mediator);
}

/// <summary>
/// 牌友A
/// </summary>
public class PartnerA : AbstractCardPartner
{
    public override void ChangeMoney(int money, AbstractMediator mediator)
    {
        mediator.ChangeMoney(money);
    }
}

/// <summary>
/// 牌友B
/// </summary>
public class PartnerB : AbstractCardPartner
{
    public override void ChangeMoney(int money, AbstractMediator mediator)
    {
        mediator.ChangeMoney(money);
    }
}

// 抽象狀態類
public abstract class State
{
    protected AbstractMediator meditor;

    public abstract void ChangeMoney(int money);
}

// 初始化狀態類
public class InitState : State
{
    public InitState()
    {
        Console.WriteLine("游戲才剛剛開始,暫時還有玩家勝出");
    }

    public override void ChangeMoney(int money)
    {
        return;
    }
}

public class AWinState : State
{
    public AWinState(AbstractMediator concretemediator)
    {
        meditor = concretemediator;
    }

    public override void ChangeMoney(int money)
    {
        foreach (AbstractCardPartner p in meditor.list)
        {
            PartnerA a = p as PartnerA;
            if (a != null)
            {
                a.Money += money;
            }
            else
            {
                p.Money -= money;
            }
        }
    }
}

public class BWinState : State
{
    public BWinState(AbstractMediator concretemediator)
    {
        meditor = concretemediator;
    }

    public override void ChangeMoney(int money)
    {
        foreach (AbstractCardPartner p in meditor.list)
        {
            PartnerB b = p as PartnerB;
            // 如果集合對象中時B對象,則對B的錢添加
            if (b != null)
            {
                b.Money += money;
            }
            else
            {
                p.Money -= money;
            }
        }
    }
}

/// <summary>
/// 抽象中介者類
/// </summary>
public abstract class AbstractMediator
{
    public List<AbstractCardPartner> list = new List<AbstractCardPartner>();

    public State State { get; set; }

    public AbstractMediator(State state)
    {
        State = state;
    }

    public void Add(AbstractCardPartner partner)
    {
        list.Add(partner);
    }

    public void Remove(AbstractCardPartner partner)
    {
        list.Remove(partner);
    }

    public void ChangeMoney(int money)
    {
        State.ChangeMoney(money);
    }
}

/// <summary>
/// 中介者Pater
/// </summary>
public class MediatorPater : AbstractMediator
{
    public MediatorPater(State state) : base(state)
    {
    }
}

/// <summary>
/// 調用
/// </summary>
/// <param name="args"></param>
static void Main(string[] args)
{
    State.Mediator.AbstractCardPartner A = new State.Mediator.PartnerA();
    State.Mediator.AbstractCardPartner B = new State.Mediator.PartnerB();
    // 初始錢
    A.Money = 20;
    B.Money = 20;

    State.Mediator.AbstractMediator mediator = new State.Mediator.MediatorPater(new State.Mediator.InitState());

    // A,B玩家進入平臺進行游戲
    mediator.Add(A);
    mediator.Add(B);

    // A 贏了
    mediator.State = new State.Mediator.AWinState(mediator);
    mediator.ChangeMoney(5);
    Console.WriteLine("A 現在的錢是:{0}", A.Money);// 應該是25
    Console.WriteLine("B 現在的錢是:{0}", B.Money); // 應該是15

    // B 贏了
    mediator.State = new State.Mediator.BWinState(mediator);
    mediator.ChangeMoney(10);
    Console.WriteLine("A 現在的錢是:{0}", A.Money);// 應該是25
    Console.WriteLine("B 現在的錢是:{0}", B.Money); // 應該是15

    Console.ReadLine();
}

狀態者模式優缺點

優點:

  • 將狀態判斷邏輯每個狀態類裡面,可以簡化判斷的邏輯。
  • 當有新的狀態出現時,可以通過添加新的狀態類來進行擴展,擴展性好。

缺點: 如果狀態過多的話,會導致有非常多的狀態類,加大了開銷

狀態者模式適用場景

在以下情況下可以考慮使用狀態者模式。

  • 當一個對象狀態轉換的條件表達式過於複雜時可以使用狀態者模式。把狀態的判斷邏輯轉移到表示不同狀態的一系列類中,可以把複雜的判斷邏輯簡單化。
  • 當一個對象行為取決於它的狀態,並且它需要在運行時刻根據狀態改變它的行為時,就可以考慮使用狀態者模式。

代碼


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

-Advertisement-
Play Games
更多相關文章
  • java特點; 1.面對象性 2.可移植性/跨平臺性 java組成; jdk(java工具開發工具包) / \ \ jre 指令集合 api和常用java包 (運行環境)(編輯器) / jvm(java虛擬器) java執行過程:java文件〉編輯器(javac)〉class文件〉jvm(解釋執行) ...
  • 題目描述 Farmer John最近為奶牛們的圖書館添置了一個巨大的書架,儘管它是如此的大,但它還是幾乎瞬間就被各種各樣的書塞滿了。現在,只有書架的頂上還留有一點空間。 所有N(1 <= N <= 20,000)頭奶牛都有一個確定的身高H_i(1 <= H_i <= 10,000)。設所有奶牛身高的 ...
  • 什麼是策略者模式? 策略模式是針對一組演算法,將每個演算法封裝到具有公共介面的獨立的類中,從而使它們可以相互替換。策略模式使得演算法可以在不影響到客戶端的情況下發生變化。對演算法的包裝,是把使用演算法的責任和演算法本身分割開,委派給不同的對象負責。策略模式通常把一系列的演算法包裝到一系列的策略類裡面。用一句話慨括 ...
  • 制定 BB 鳥規則時需遵循的規範: • 只包含不可變的規則,而不是籠統的說明• 總是把規則提煉成最簡單的表達• 總是首先說明規則是什麼,再說明“如果不這樣,那麼會如何”• 每個規則必須包含以下詞中的一個——總是、永遠不要、只有、每一個、不要、要 設計系統的規則列表: • 永遠不要給佈局的子內容強加內 ...
  • 1.- DRY: Don’t repeat yourself. DRY 是一個最簡單的法則,也是最容易被理解的。但它也可能是最難被應用的(因為要做到這樣,我們需要在泛型設計上做相當的努力,這並不是一件容易的事)。它意味著,當我們在兩個或多個地方的時候發現一些相似的代碼的時候,我們需要把他們的共性抽象 ...
  • 這幾天在知乎看到了許多關於濫用三層的討論,很多觀點都暗指三層架構這東西太Low了。Low嗎?其實不然。我們覺得三層Low主要是因為三層太簡單了,人們往往不理解三層的思想和目的,濫用三層,加之一些劣質程式員為其加上些自認牛逼的更改搞個五六七八層,代碼寫得又爛,結果看起來就很Low,於是人們就把鍋扣到三 ...
  • 進入工作室也有一段時間了,期間也進行過多次授課,使我對c#這門新的編程語言有了一定的理解。 c#是一種純面向對象的程式設計語言,設計一個程式就是設計一個或多個類,這與之前我所學習的c和c++有些不一樣,它們都是通過函數來實驗目標,而c#是通過方法來完成的。c#中的方法是在類或結構中定義的,有點類似於 ...
  • dubbo支持多種遠程調用方式,例如dubbo RPC(二進位序列化 + tcp協議)、http invoker(二進位序列化 + http協議,至少在開源版本沒發現對文本序列化的支持)、hessian(二進位序列化 + http協議)、WebServices (文本序列化 + http協議)等等,... ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...