深入解析策略模式

来源:http://www.cnblogs.com/lewis0077/archive/2016/01/25/5133812.html
-Advertisement-
Play Games

在講策略模式之前,我們先看一個日常生活中的小例子: 現實生活中我們到商場買東西的時候,賣場往往根據不同的客戶制定不同的報價策略,比如針對新客戶不打折扣,針對老客戶打9折,針對VIP客戶打8折... 現在我們要做一個報價管理的模塊,簡要點就是要針對不同的客戶,提供不同的折扣報價。如果是有你來做...


  在講策略模式之前,我們先看一個日常生活中的小例子:

  現實生活中我們到商場買東西的時候,賣場往往根據不同的客戶制定不同的報價策略,比如針對新客戶不打折扣,針對老客戶打9折,針對VIP客戶打8折...

  現在我們要做一個報價管理的模塊,簡要點就是要針對不同的客戶,提供不同的折扣報價。

如果是有你來做,你會怎麼做?

我們很有可能寫出下麵的代碼:

package strategy.examp02;

import java.math.BigDecimal;

public class QuoteManager {

    public BigDecimal quote(BigDecimal originalPrice,String customType){
        if ("新客戶".equals(customType)) {
            System.out.println("抱歉!新客戶沒有折扣!");
            return originalPrice;
        }else if ("老客戶".equals(customType)) {
            System.out.println("恭喜你!老客戶打9折!");
            originalPrice = originalPrice.multiply(new BigDecimal(0.9)).setScale(2,BigDecimal.ROUND_HALF_UP);
            return originalPrice;
        }else if("VIP客戶".equals(customType)){
            System.out.println("恭喜你!VIP客戶打8折!");
            originalPrice = originalPrice.multiply(new BigDecimal(0.8)).setScale(2,BigDecimal.ROUND_HALF_UP);
            return originalPrice;
        }
        //其他人員都是原價
        return originalPrice;
    }

}

經過測試,上面的代碼工作的很好,可是上面的代碼是有問題的。上面存在的問題:把不同客戶的報價的演算法都放在了同一個方法裡面,使得該方法很是龐大(現在是只是一個演示,所以看起來還不是很臃腫)。

下麵看一下上面的改進,我們把不同客戶的報價的演算法都單獨作為一個方法

 

 1 package strategy.examp02;
 2 
 3 import java.math.BigDecimal;
 4 
 5 public class QuoteManagerImprove {
 6 
 7     public BigDecimal quote(BigDecimal originalPrice, String customType){
 8         if ("新客戶".equals(customType)) {
 9             return this.quoteNewCustomer(originalPrice);
10         }else if ("老客戶".equals(customType)) {
11             return this.quoteOldCustomer(originalPrice);
12         }else if("VIP客戶".equals(customType)){
13             return this.quoteVIPCustomer(originalPrice);
14         }
15         //其他人員都是原價
16         return originalPrice;
17     }
18 
19     /**
20      * 對VIP客戶的報價演算法
21      * @param originalPrice 原價
22      * @return 折後價
23      */
24     private BigDecimal quoteVIPCustomer(BigDecimal originalPrice) {
25         System.out.println("恭喜!VIP客戶打8折");
26         originalPrice = originalPrice.multiply(new BigDecimal(0.8)).setScale(2,BigDecimal.ROUND_HALF_UP);
27         return originalPrice;
28     }
29 
30     /**
31      * 對老客戶的報價演算法
32      * @param originalPrice 原價
33      * @return 折後價
34      */
35     private BigDecimal quoteOldCustomer(BigDecimal originalPrice) {
36         System.out.println("恭喜!老客戶打9折");
37         originalPrice = originalPrice.multiply(new BigDecimal(0.9)).setScale(2,BigDecimal.ROUND_HALF_UP);
38         return originalPrice;
39     }
40 
41     /**
42      * 對新客戶的報價演算法
43      * @param originalPrice 原價
44      * @return 折後價
45      */
46     private BigDecimal quoteNewCustomer(BigDecimal originalPrice) {
47         System.out.println("抱歉!新客戶沒有折扣!");
48         return originalPrice;
49     }
50 
51 }

 

上面的代碼比剛開始的時候要好一點,它把每個具體的演算法都單獨抽出來作為一個方法,當某一個具體的演算法有了變動的時候,只需要修改響應的報價演算法就可以了。

但是改進後的代碼還是有問題的,那有什麼問題呢?

1.當我們新增一個客戶類型的時候,首先要添加一個該種客戶類型的報價演算法方法,然後再quote方法中再加一個else if的分支,是不是感覺很是麻煩呢?而且這也違反了設計原則之一的開閉原則(open-closed-principle).

開閉原則:

  對於擴展是開放的(Open for extension)。這意味著模塊的行為是可以擴展的。當應用的需求改變時,我們可以對模塊進行擴展,使其具有滿足那些改變的新行為。也就是說,我們可以改變模塊的功能。

  對於修改是關閉的(Closed for modification)。對模塊行為進行擴展時,不必改動模塊的源代碼或者二進位代碼。

2.我們經常會面臨這樣的情況,不同的時期使用不同的報價規則,比如在各個節假日舉行的各種促銷活動時、商場店慶時往往都有普遍的折扣,但是促銷時間一旦過去,報價就要回到正常價格上來。按照上面的代碼我們就得修改if else裡面的代  碼很是麻煩

那有沒有什麼辦法使得我們的報價管理即可擴展、可維護,又可以方便的響應變化呢?當然有解決方案啦,就是我們下麵要講的策略模式。

 

定義:

  策略模式定義了一系列的演算法,並將每一個演算法封裝起來,使每個演算法可以相互替代,使演算法本身和使用演算法的客戶端分割開來,相互獨立。

結構:

  1.策略介面角色IStrategy:用來約束一系列具體的策略演算法,策略上下文角色ConcreteStrategy使用此策略介面來調用具體的策略所實現的演算法。

  2.具體策略實現角色ConcreteStrategy:具體的策略實現,即具體的演算法實現。

  3.策略上下文角色StrategyContext:策略上下文,負責和具體的策略實現交互,通常策略上下文對象會持有一個真正的策略實現對象,策略上下文還可以讓具體的策略實現從其中獲取相關數據,回調策略上下文對象的方法。

UML類圖:

                                                   

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

-Advertisement-
Play Games
更多相關文章
  • 利用Python語言實現Grib數據可視化主要依靠三個庫——pygrib、numpy和matplotlib。pygrib是歐洲中期天氣預報中心(ECMWF)的GRIG API C庫的Python介面,通過這個庫可以將Grib數據讀取出來;numpy是Python的一種開源的數值計算擴展,這種工具可用...
  • 上述if語句的等值判斷,可以用switch來代替。註意每個case後面一般要添加break,表示當前這個case執行完了;防止出現case穿透,即繼續執行case,直到遇到break才跳出。下麵例子反過來利用了case穿透現象。【例子】JDK7.0新特性:增強switch在JDK7之前,switch...
  • #!/bin/env python # -*- coding: UTF-8 -*- # 必須以root許可權運行 import socket import sys import timeimport random from struct import * # 計算校驗和 def ...
  • 回到目錄dynamic這個動態類型早在.net3.5時就已經出現了,當時是伴隨的Linq一起讓我們認識的,但在使用時總覺得有點彆扭,因為它是internal的,所以不能跨程式集使用,這對於分層開發的我們來說顯然是不能接受的,所以把dynamic了冷落了很久,應該說是5年吧,哈哈,這幾天在睡覺時,突然...
  • 值對象(value object)===================== 什麼是值對象 維基百科的定義 In computer science, a value object is a small object that represents a simple entity whose equ....
  • 序言在上一篇配置iis負載均衡中我們使用啦微軟的ARR,我在那篇文章也中提到了網站的高可用性,但是ARR只能做請求入口的消息分發服務,這樣如果我們的消息分發伺服器給down掉啦,那麼做再多的應用服務集群也都枉然。這篇文章我主要針對解決這一問題來做分析,引入NLB,相對於ARR來說,ARR算是應用級別...
  • ---單例設計模式之餓漢式--- 創建SingleInstance類 1 /** 2 * 單例設計模式之餓漢式 3 */ 4 public class SingleInstance { 5 /** 6 * 私有化構造方法 7 */ 8 priv...
  • 在前面搭建的基礎上,引入新的jar包如下:aopalliance-1.0.jaraspectjweaver-1.8.8.jarmybatis-3.3.0.jarmybatis-spring-1.2.3.jarmysql-connector-java-5.1.31.jarspring-aop-4.2....
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...