自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

設(shè)計(jì)模式中的策略模式與狀態(tài)模式

開(kāi)發(fā) 架構(gòu)
在軟件設(shè)計(jì)模式中有兩種模式很相似,它們就是狀態(tài)模式和策略模式。本文將為您分析他們的優(yōu)缺點(diǎn)。

使用場(chǎng)景

狀態(tài)模式:當(dāng)對(duì)象的行為隨對(duì)象的狀態(tài)的改變而改變時(shí),我們?yōu)榱私怦疃嘀嘏袛鄺l件,封裝行為的變化,可以定義一個(gè)抽象的狀態(tài)類,提供對(duì)象行為接口。具體與狀態(tài)相關(guān)的行為,由它的子類去實(shí)現(xiàn)。

策略模式:“策略”二字等同于算法,當(dāng)現(xiàn)實(shí)系統(tǒng)中需要對(duì)算法動(dòng)態(tài)指定,并且可以互相替換,可以抽象出算法的調(diào)用接口,具體的算法實(shí)現(xiàn)由具體的策略角色去實(shí)現(xiàn),根據(jù)里氏替換原則,任何出現(xiàn)父類的地方都可以使用它的子類去替換,這樣符合我們的業(yè)務(wù)需求。

比較

雖然兩者都是抽象出父類規(guī)范調(diào)用接口,具體的行為由子類實(shí)現(xiàn),環(huán)境對(duì)象同時(shí)包含父類的引用,但是這兩者模式應(yīng)用的場(chǎng)景完全不同。例如:你去ATM機(jī)取款,如果你的賬戶處于凍結(jié)狀態(tài),就不能正常取款。這里你的銀行賬戶至少有兩個(gè)狀態(tài)就是凍結(jié)與非凍結(jié)。這樣的業(yè)務(wù)解決方案你應(yīng)該不會(huì)想到用策略模式去解決。又如商場(chǎng)打折,有很多打折的策略,兒童用品打7折,老人用品打5折,這個(gè)問(wèn)題不會(huì)跟狀態(tài)模式扯上關(guān)系吧,這里商品你不能說(shuō)有老人狀態(tài)和兒童狀態(tài)!

ATM取款案例分析

設(shè)計(jì)到的角色:取款人、賬戶、ATM機(jī)。

用例圖

這邊簡(jiǎn)單起見(jiàn),就分析取款用例。

基本事件路徑

(1) 取款人插入銀行卡,輸入密碼;

(2) 選擇取款操作,輸入取款金額;

(3) 等待出鈔,取款。

在這個(gè)用例中,如果取款人賬戶已凍結(jié),就發(fā)生一個(gè)可選事件路徑,取款用例就被終止。

關(guān)鍵的業(yè)務(wù)需求是用戶取款事實(shí),所以領(lǐng)域建??梢詮臉?biāo)識(shí)取款和取款人這兩個(gè)類開(kāi)始。這邊考慮到取款人不僅有取款這個(gè)用例,我們將取款泛化成交易。

建模

用例實(shí)化

取款:基本事件路徑的實(shí)化。

細(xì)化領(lǐng)域模型

這邊引入了ATMSystem這個(gè)邊界對(duì)象作為控制器負(fù)責(zé)與用戶交互(與用戶交互的ATM操作界面),Trade是負(fù)責(zé)處理用戶交易的類。

取款人插入銀行卡輸入密碼,ATMSystem負(fù)責(zé)將驗(yàn)證賬戶信息的消息傳遞給Trade交易類,Trade根據(jù)傳過(guò)來(lái)的賬戶密碼驗(yàn)證用戶是否存在,然后將處理結(jié)束的消息傳遞給ATMSystem這個(gè)邊界對(duì)象反映給用戶。

分析模型中的一些類

取款設(shè)計(jì)——應(yīng)用狀態(tài)模式

賬戶的狀態(tài)分為凍結(jié)狀態(tài)和激活狀態(tài)。賬戶在激活的狀態(tài)下可以進(jìn)行交易,而在凍結(jié)狀態(tài)下不能做任何交易。在賬戶驗(yàn)證的時(shí)候,應(yīng)該返回當(dāng)前賬戶的狀態(tài),如果賬戶處于凍結(jié)狀態(tài),當(dāng)取款人進(jìn)行取款操作的時(shí)候直接返回錯(cuò)誤提示信息。狀態(tài)圖表示如下:

取款應(yīng)用狀態(tài)模式結(jié)構(gòu)圖:

在驗(yàn)證賬戶階段,如果賬戶處于凍結(jié)狀態(tài),則直接返回DeadAccount謝絕一切交易,否則返回ActiveAccount處理后面的交易。

示意代碼:

這邊為了簡(jiǎn)單實(shí)現(xiàn),省去了ATMSystem這個(gè)邊界對(duì)象。

用戶賬號(hào)信息抽象類

  1. /// <summary> 
  2. /// 用戶賬號(hào)信息 
  3. /// </summary> 
  4. public abstract class Account 
  5.     /// <summary> 
  6.     /// 賬號(hào) 
  7.     /// </summary> 
  8.     private string account; 
  9.     /// <summary> 
  10.     /// 密碼 
  11.     /// </summary> 
  12.     private string pwd; 
  13.     public abstract decimal getBalance(decimal d); 

凍結(jié)賬號(hào)類

  1. /// <summary> 
  2. /// 凍結(jié)賬戶類 
  3. /// </summary> 
  4. public class DeadAccount:Account 
  5.   
  6.     public override decimal getBalance(decimal d) 
  7.     { 
  8.         return 0; 
  9.     } 

激活賬戶類

  1. /// <summary> 
  2. /// 激活賬戶類 
  3. /// </summary> 
  4. public class ActiveAccount:Account 
  5.     public override decimal getBalance(decimal d) 
  6.     { 
  7.         return d; 
  8.     } 

交易類

  1. /// <summary> 
  2. /// 交易類 負(fù)責(zé)用戶交易處理的類 
  3. /// </summary> 
  4. public class Trade 
  5.     /// <summary> 
  6.     /// 保存用戶賬號(hào)信息 
  7.     /// </summary> 
  8.     private Account account; 
  9.     public Account VolidateLogin(Account a) 
  10.     {            
  11.         //query the database to validate the user exists 
  12.         //For Example 
  13.         this.account = new DeadAccount(); 
  14.         return this.account; 
  15.     } 
  16.   
  17.     /// <summary> 
  18.     /// 商業(yè)邏輯 取款 
  19.     /// </summary> 
  20.     /// <param name="d"></param> 
  21.     /// <returns></returns> 
  22.     public decimal GetBalance(decimal d) 
  23.     { 
  24.         return this.account.getBalance(d); 
  25.     } 
  26.  } 

用戶類

  1. /// <summary> 
  2. /// 取款人 
  3. /// </summary> 
  4. public class User 
  5.     /// <summary> 
  6.     /// 用戶賬號(hào)信息 
  7.     /// </summary> 
  8.     private Account account; 
  9.     /// <summary> 
  10.     /// 交易處理類 
  11.     /// </summary> 
  12.     private Trade trade; 
  13.   
  14.     public User(Account a, Trade t) 
  15.     { 
  16.         this.account = a; 
  17.         tthis.trade = t; 
  18.     } 
  19.     /// <summary> 
  20.     /// 用戶登錄類 
  21.     /// </summary> 
  22.     public void Login() 
  23.     { 
  24.         trade.VolidateLogin(account); 
  25.     } 
  26.   
  27.     /// <summary> 
  28.     /// 取款 
  29.     /// </summary> 
  30.     /// <param name="d"></param> 
  31.     /// <returns></returns> 
  32.     public decimal GetBalance(decimal d) 
  33.     { 
  34.         return trade.GetBalance(d); 
  35.     } 

客戶端代碼

  1. class Client 
  2.     { 
  3.         static void Main(string[] args) 
  4.         { 
  5.             //開(kāi)始用戶取款,默認(rèn)是激活賬戶 
  6.             ActiveAccount aa = new ActiveAccount(); 
  7.             Trade t = new Trade(); 
  8.             User u = new User(aa,t); 
  9.             //先登錄,后取款 
  10.             u.Login(); 
  11.             Console.WriteLine(u.GetBalance(100)); 
  12.             Console.ReadLine(); 
  13.         } 
  14.     } 

用戶必須先登錄(插入銀行卡,輸入密碼,點(diǎn)確定),才能選擇取款業(yè)務(wù)(選擇取款)。登錄之后,返回的賬號(hào)對(duì)象作為trade類成員,當(dāng)進(jìn)行取款業(yè)務(wù)的時(shí)候直接引用該賬號(hào)成員獲得取款信息。如果該賬號(hào)屬于凍結(jié)賬號(hào),則直接返回0。否則返回取款的值。

商品折扣案例

案例描述:某家超市國(guó)慶節(jié)為了促銷,某些類商品打折,比如運(yùn)動(dòng)鞋打8折、秋裝打9折等,張三去購(gòu)物為了一雙運(yùn)動(dòng)鞋、一件秋裝、一瓶洗發(fā)水。。。,張三買完?yáng)|西回家,心想今天自己總共“賺”了多少錢? 

案例分析:商家對(duì)于商品打折可能有很多策略,這里使用策略模式,封裝商品打折策略,這樣以便以后擴(kuò)展了打折策略,不用去修改原來(lái)的代碼,具有很好的靈活性。

模式涉及的角色: 

抽象策略角色:通常由一個(gè)接口或者抽象實(shí)現(xiàn);

具體策略角色:包裝了相關(guān)的算法和行為;

環(huán)境角色:包含抽象策略角色的引用,最終供用戶使用。

商品折扣案例設(shè)計(jì)圖

顧客購(gòu)物涉及到的角色有:購(gòu)物車、商品、收銀臺(tái)、抽象策略角色和具體策略角色。

購(gòu)物車:是擺放商品的容器,提供添加和刪除操作;

商品:商品實(shí)體,有類型,商品名、價(jià)格等屬性;

收銀臺(tái):負(fù)責(zé)收錢,主要是計(jì)算顧客購(gòu)買所有商品的價(jià)格和折扣總額;

抽象策略角色:提供折扣策略接口。

具體策略角色:實(shí)現(xiàn)具體折扣算法。

商品折扣示意代碼:

 
  1. /// <summary> 
  2. /// 具體商品類 
  3. /// </summary> 
  4. public class goods 
  5.     /// <summary> 
  6.     /// 商品類型 
  7.     /// </summary> 
  8.     public string Type 
  9.     { 
  10.         set; 
  11.         get; 
  12.     } 
  13.   
  14.     /// <summary> 
  15.     /// 商品名稱 
  16.     /// </summary> 
  17.     public string Name 
  18.     { 
  19.         get; 
  20.         set; 
  21.     } 
  22.   
  23.   
  24.     /// <summary> 
  25.     /// 商品價(jià)格 
  26.     /// </summary> 
  27.     public decimal Price 
  28.     { 
  29.         get; 
  30.         set; 
  31.     } 
  32.   

抽象策略角色

  1. /// <summary> 
  2. /// 抽象策略接口 
  3. /// </summary> 
  4. public interface IDiscountStrategy 
  5.     decimal GetDiscount(goods g); 

具體策略角色

 

  1. /// <summary> 
  2. /// 秋裝打折策略 
  3. /// </summary> 
  4. public class AutumnDressDiscountStrategy:IDiscountStrategy 
  5.     #region IDiscountStrategy Members 
  6.   
  7.     public decimal GetDiscount(goods g) 
  8.     { 
  9.         return (decimal)0.9 * g.Price; 
  10.     } 
  11.   
  12.     #endregion 
  13.   
  14.   
  15. /// <summary> 
  16. /// 運(yùn)動(dòng)鞋打折策略 
  17. /// </summary> 
  18. public class SportShoesDiscountStrategy:IDiscountStrategy 
  19.     #region IDiscountStrategy Members 
  20.   
  21.     public decimal GetDiscount(goods g) 
  22.     { 
  23.         return g.Price * (decimal)0.8; 
  24.     } 
  25.   
  26.     #endregion 

購(gòu)物車

 
  1. /// <summary> 
  2. /// 購(gòu)物車類 負(fù)責(zé)商品的維護(hù) 
  3. /// </summary> 
  4. public class ShoppingCar 
  5.     private List<goods> goodsList=new List<goods>(); 
  6.   
  7.     /// <summary> 
  8.     /// 將商品加入到購(gòu)物車 
  9.     /// </summary> 
  10.     /// <param name="g"></param> 
  11.     public void AddGoods(goods g) 
  12.     { 
  13.         goodsList.Add(g); 
  14.     } 
  15.   
  16.     /// <summary> 
  17.     /// 將商品從購(gòu)物車當(dāng)中移除 
  18.     /// </summary> 
  19.     /// <param name="g"></param> 
  20.     public void RemoveGoods(goods g) 
  21.     { 
  22.         goodsList.Remove(g); 
  23.     } 
  24.   
  25.     public List<goods> GoodsList 
  26.     { 
  27.         get 
  28.         { 
  29.             return goodsList; 
  30.         } 
  31.     } 
  32.   

收銀臺(tái)角色

 

  1. /// <summary> 
  2. /// 收銀臺(tái) 
  3. /// </summary> 
  4. public class CashierDesk 
  5.     /// <summary> 
  6.     /// 購(gòu)物車 
  7.     /// </summary> 
  8.     private ShoppingCar shoppingCar; 
  9.     /// <summary> 
  10.     /// 策略字典 
  11.     /// </summary> 
  12.     private Dictionary<string, IDiscountStrategy> strategies; 
  13.   
  14.     public CashierDesk(ShoppingCar sc, Dictionary<string, IDiscountStrategy> s) 
  15.     { 
  16.         this.shoppingCar = sc
  17.         this.strategies = s; 
  18.     } 
  19.   
  20.     /// <summary> 
  21.     /// 獲得所有商品的價(jià)格 
  22.     /// </summary> 
  23.     /// <returns></returns> 
  24.     public decimal GetTotalPrice() 
  25.     { 
  26.         return shoppingCar.GoodsList.Sum(p => p.Price); 
  27.     } 
  28.   
  29.     /// <summary> 
  30.     /// 獲得所有商品的總的折扣 
  31.     /// </summary> 
  32.     /// <returns></returns> 
  33.     public decimal GetTotalDiscount() 
  34.     { 
  35.         decimal sum = 0
  36.         IDiscountStrategy idiscountStrategy; 
  37.         foreach (goods g in shoppingCar.GoodsList) 
  38.         { 
  39.             idiscountStrategy=strategies.SingleOrDefault(p => p.Key == g.Type).Value; 
  40.             if (idiscountStrategy != null) 
  41.             { 
  42.                 sum += idiscountStrategy.GetDiscount(g); 
  43.             } 
  44.         } 
  45.         return sum; 
  46.     } 

客戶端代碼

 
  1. class Client 
  2.     { 
  3.         static void Main(string[] args) 
  4.         { 
  5.             ShoppingCar sc = new ShoppingCar(); 
  6.             Dictionary<string, IDiscountStrategy> discountD = new Dictionary<string, IDiscountStrategy>(); 
  7.             //向購(gòu)物車中加入商品 
  8.             sc.AddGoods(new goods {Name="NIKE鞋 ",Price=100,Type="運(yùn)動(dòng)鞋" }); 
  9.             sc.AddGoods(new goods { Name = "秋裝"Price = 200Type = "秋裝" }); 
  10.             sc.AddGoods(new goods { Name = "蘋(píng)果"Price = 300Type = "水果" }); 
  11.             //配置折扣策略 
  12.             discountD.Add("運(yùn)動(dòng)鞋", new SportShoesDiscountStrategy()); 
  13.             discountD.Add("秋裝", new AutumnDressDiscountStrategy()); 
  14.             CashierDesk cd = new CashierDesk(sc, discountD); 
  15.             //得到所有商品總價(jià) 
  16.             Console.WriteLine(cd.GetTotalPrice()); 
  17.             //得到所有商品折扣價(jià) 
  18.             Console.WriteLine(cd.GetTotalDiscount()); 
  19.   
  20.             Console.ReadLine(); 
  21.         } 
  22.     } 

策略模式優(yōu)點(diǎn)與缺點(diǎn):

優(yōu)點(diǎn)

封裝了算法不穩(wěn)定性,易于以后業(yè)務(wù)策略的擴(kuò)展。

缺點(diǎn)

每種策略對(duì)應(yīng)于一個(gè)具體策略角色類,會(huì)增加系統(tǒng)需要維護(hù)的類的數(shù)量。

 

原文鏈接:http://www.cnblogs.com/ejiyuan/archive/2012/06/28/2567905.html

【編輯推薦】

 

 

責(zé)任編輯:彭凡 來(lái)源: 博客園
相關(guān)推薦

2013-11-26 16:09:34

Android設(shè)計(jì)模式

2014-12-29 10:39:16

JS

2020-11-04 08:54:54

狀態(tài)模式

2015-09-08 13:39:10

JavaScript設(shè)計(jì)模式

2021-06-09 08:53:34

設(shè)計(jì)模式策略模式工廠模式

2017-07-07 10:55:14

數(shù)據(jù)庫(kù)MongoDB設(shè)計(jì)模式

2024-01-29 12:22:07

設(shè)計(jì)模式策略模式

2024-10-14 08:39:29

工廠模式策略模式代碼

2024-06-19 16:02:46

2024-10-06 12:56:36

Golang策略設(shè)計(jì)模式

2021-04-19 21:25:48

設(shè)計(jì)模式到元

2021-04-18 21:07:32

門面模式設(shè)計(jì)

2024-07-31 08:12:33

2010-01-21 09:08:53

.NET設(shè)計(jì)模式

2021-04-14 09:02:22

模式 設(shè)計(jì)建造者

2021-02-01 10:01:58

設(shè)計(jì)模式 Java單例模式

2024-12-09 09:40:00

策略模式Java

2024-07-22 08:09:28

C#模式架構(gòu)

2022-01-12 13:33:25

工廠模式設(shè)計(jì)

2023-05-04 08:47:31

命令模式抽象接口
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)