記一場精彩的籃球比賽——淺談策略模式

来源:https://www.cnblogs.com/MrJR/archive/2019/02/27/10441166.html
-Advertisement-
Play Games

策略模式 雖然我本人比較討厭一些很官方的術語定義,因為我經常弄不明白有些定義講了個啥,但是為了讓這篇博文顯得不那麼輕浮,所以我也就不能免俗的先將設計模式之策略模式的定義首先丟到各位看官面前。 策略模式定義了演算法族,分別封裝起來,讓他們之間可以互相替換,此模式讓演算法的變化獨立於使用演算法的客戶。 第一眼 ...


策略模式

聲明:本文為原創,如有轉載請註明轉載與原作者並提供原文鏈接,僅為學習交流,本人才識短淺,如有錯誤,敬請指正

雖然我本人比較討厭一些很官方的術語定義,因為我經常弄不明白有些定義講了個啥,但是為了讓這篇博文顯得不那麼輕浮,所以我也就不能免俗的先將設計模式之策略模式的定義首先丟到各位看官面前。

策略模式定義了演算法族,分別封裝起來,讓他們之間可以互相替換,此模式讓演算法的變化獨立於使用演算法的客戶。

第一眼這個定義看上去,難免會讓人心生膽怯,又是演算法族,又是封裝,又是替換,似乎很複雜的樣子,但是實際上,很好理解,不信,我們來一起看一場非常激烈的籃球比賽吧,相信看完這場比賽之後你就能大致掌握這個設計模式了。

PS:有人肯定要問了,為啥沒有個UML圖呢,其他人說設計模式都會咣當扔個UML圖上來,你咋沒有呢,是不是偷工減料,其實啊,真不是,在於我這個人呢,比較喜歡直接擼代碼看代碼,也沒有說看UML圖理解一個設計模式的,並不說UML圖不重要,只是策略模式的UML圖在網上一搜一大把,我還是不要當搬運工,複製粘貼了吧。

歡迎大家來到NBA賽場,今天的比賽呢,雙方分別是湖人和騎士,科比與詹姆斯的宿命對決,作為籃球運動員,在場上,最重要的兩件事是啥呢,沒錯,就是“投籃”和“傳球”,投籃是為了得分,傳球是為了更輕鬆的投籃得分,投籃有很多種方式,後仰投籃,三分遠投,急停跳投,傳球包括擊地傳球,不看人傳球等,而在Java的設計模式中,我們可以把這些統統定義成“方法”。

我們首先抽象出來一個投籃相關的介面,它包含了一個方法:shoot(),即投籃

public interface ShootStrategy {

    public void shoot();
}

同樣的,我們抽象出來一個傳球相關的介面,它包含了一個方法:pass(),即傳球

public interface PassStrategy {

    public void pass();
}

那麼這兩個介面有什麼用呢,回到策略模式的定義,註意“演算法族”這三個字,什麼是演算法族,如果後仰投籃是一個演算法,三分遠投是一個演算法,急停跳投也是一個演算法,我們就會發現他們的共同點是,他們都是投籃的演算法,也就是說他們都是投籃的演算法族,同理,擊地傳球與不看人傳球也是傳球的演算法族,而後策略模式的定義說要分別封裝他們,提到封裝大家想到了什麼呢,類!

接下來我們嘗試封裝一下後仰投籃演算法和三分遠投演算法

public class BackwardShoot implements ShootStrategy {

    @Override
    public void shoot() {
        System.out.println("標誌性的後仰跳投");
    }
}

封裝後仰跳投演算法,代碼很簡單,因為我們不希望演算法的邏輯干擾到我們著眼於真正的重點——策略模式。

public class ThreeShoot implements ShootStrategy {

    @Override
    public void shoot() {

        System.out.println("神準的三分");
    }
}

封裝三分遠投演算法。

同樣的,我們也封裝一下傳球的演算法,畢竟籃球離不開傳球(科比:傳球,傳球是什麼意思,誤~)

public class DropPass implements PassStrategy {

    @Override
    public void pass() {
        System.out.println("詭異的擊地傳球");
    }
}

封裝擊地傳球演算法

public class NoLookPass implements PassStrategy {

    @Override
    public void pass() {
        System.out.println("一記精彩的不看人傳球");
    }
}

封裝不看人傳球演算法

好了,現在我們所有的策略都準備好了(原諒我這裡直接使用了策略,不用驚嚇,我們剛剛已經完成了策略模式最重要的部分),就差我們的運動員上場了。

我們首先定義一個籃球運動員的類。

public class BasketballPlayer {

    private String name;

    private ShootStrategy shootStrategy;
    private PassStrategy passStrategy;

    public void shoot(){
        System.out.println("------" + name + "------");
        shootStrategy.shoot();
    }

    public void pass(){
        System.out.println("------" + name + "------");
        passStrategy.pass();
    }

    public void setShootStrategy(ShootStrategy shootStrategy) {
        this.shootStrategy = shootStrategy;
    }

    public void setPassStrategy(PassStrategy passStrategy) {
        this.passStrategy = passStrategy;
    }

    public void selfIntroduction(){
        System.out.println("我是個籃球運動員");
    }

    public void setName(String name) {
        this.name = name;
    }
}

看一下這段代碼,我定義了三個變數,name,表示運動員的名字,然後是我自己定義的投籃介面以及傳球介面,這倆乾乾巴巴,麻麻賴賴的介面放這裡有啥用呢,盤他,我們往下看,我們會發現,籃球運動員的投籃,使用的是我們投籃介面里的shoot()方法實現,而傳球呢,使用的是我們傳球介面里的pass()方法實現,有人就會想了,我這倆介面都沒方法體,調用個毛啊,稍安勿躁,繼續往下看,下麵是兩個平平無奇的setter方法,然而,奧秘就是在這裡,這個奧秘,我們一般叫他“解耦”,通過使用Java的介面回調,我們可以將任意對象傳入這個類中供這個類使用,只要這個類實現了對應的介面即可,在這裡就是投籃介面與傳球介面,然後我們就會發現,剛纔我們實現的後仰投籃,三分遠投等,這裡統統受用!

說了之久,我們的大明星怎麼還沒露面呢,繼續

科比是一名(is-a)籃球運動員,所以

public class Kobe extends BasketballPlayer {

    @Override
    public void selfIntroduction(){
        System.out.println("我的名字是科比");
    }
}

詹姆斯是一名(is-a)籃球運動員,所以

public class James extends BasketballPlayer {

    @Override
    public void selfIntroduction(){
        System.out.println("我叫詹姆斯");
    }
}

註意,雖然兩個類代碼比較少,但是不要忘了,由於他倆繼承了籃球運動員類,所以籃球運動員里的那些方法,他們也都擁有。

好啦,現在萬事俱備,讓我們開始這一場激動人心的比賽吧

public class LakerVsCavalier {

    public static void main(String[] args) {

        Kobe kobe = new Kobe();
        kobe.selfIntroduction();
        kobe.setName("科比");

        James james = new James();
        james.selfIntroduction();
        james.setName("詹姆斯");

        kobe.setPassStrategy(new NoLookPass());
        kobe.setShootStrategy(new BackwardShoot());
        kobe.pass();
        kobe.shoot();

        james.setPassStrategy(new DropPass());
        james.setShootStrategy(new ThreeShoot());
        james.pass();
        james.shoot();

        kobe.setPassStrategy(new DropPass());
        kobe.setShootStrategy(new ThreeShoot());
        kobe.pass();
        kobe.shoot();

        james.setPassStrategy(new NoLookPass());
        james.setShootStrategy(new BackwardShoot());
        james.pass();
        james.shoot();
    }
}

 

雖然代碼看起來很簡單,只是讓科比與詹姆斯兩個對象shoot()與pass()而已,然而事情其實並沒有那麼簡單,回顧策略模式的定義,“讓它們之間可以互相替換”,這就是問題的關鍵,我們為同一個對象設置了不同的演算法,科比可以先設置傳球策略為不看人傳球,設置投籃策略為後仰跳投,然後此封裝的演算法就會被科比對象中的pass()和shoot()方法調用,原因上文已述,然後,我們可以重新為科比對象設置不同的演算法,我們把傳球策略為擊地傳球,投籃策略設置為三分遠投,然後科比再次傳球(調用pass方法),就變成了擊地傳球,再次投籃(調用shoot方法),就變成了三分遠投,詹姆斯對象同理。

我們來看一下執行結果

一切皆如我們所料。

至此,我們也就能理解策略模式定義的最後一句:此模式讓演算法的變化獨立於使用演算法的客戶。

更重要的一點是,通過策略模式,我們真正的將易於變化的部分抽出來,使代碼對修改關閉,對擴展開放,假設我們要添加一個新的策略——急停跳投,我們只需要通過實現投籃介面的方式新建一個急停跳投策略,然後把他set進科比,詹姆斯這種使用演算法的對象即可,完全不用修改已有的調用演算法的代碼!

有興趣的可以自己去實現看看哦

今天的籃球比賽轉播到此結束,我是設計模式評論員JR,我們下次再見。


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

-Advertisement-
Play Games
更多相關文章
  • 1.什麼是命名空間,官方文檔定義為: 什麼是命名空間?從廣義上來說,命名空間是一種封裝事物的方法。在很多地方都可以見到這種抽象概念。例如,在操作系統中目錄用來將相關文件分組,對於目錄中的文件來說,它就扮演了命名空間的角色。具體舉個例子,文件 foo.txt 可以同時在目錄/home/greg 和 / ...
  • 摘要 之前老是聽說動態代理,一直沒有機會好好看過,現在就把動態代理的實現邏輯和用處整理一下。首先提兩個概念,委托類和代理類。委托類就是實際業務邏輯的處理者,代理類是處於請求發起者與委托類之間的角色,所有對委托類的請求都會經過代理類。就是委托類將請求處理委托給代理類,代理類可以起到方法攔截、功能增強的 ...
  • 一、背景 程式的定義:程式=數據+演算法+介面 二、常用技巧 技巧1 - 按目標設計介面做冪等設計 - 場景 背景:做任務賺積分。前端發出增加積分請求,如果收不到響應會重試。 後臺開發人員:怎麼判斷是重試還是另一次請求? 解決方案:介面定義中需要傳入原來積分是多少,增加到多少。開發人員直接將目標結果入 ...
  • 題意 "題目鏈接" Sol 這題就是一個很顯然的貪心。。。 首先二分一個答案,然後check是否可行。check的時候我們需要對每個位置$i$,維護出所有左端點在$i$左側,右端點在$i$右側的所有區間。最優策略一定是加右端點最遠的。 然後就做完了, 複雜度$O(nlogn)$ cpp includ ...
  • 1111 ...
  • 11 ...
  • 一、綜述 類是我們自己定義的數據類型(新類型) 設計類時要考慮的角度: (1)站在設計和實現者的角度來考慮 (2)站在使用者的角度來考慮 (3)父類,子類 二、類基礎 (1)一個類就是一個用戶自己定義的數據類型,把類可以想象為一個命名空間,包著一堆東西(成員函數,成員變數)。 (2)一個類的構成:成 ...
  • 人生苦短,我用 Python —— Life is short, you need Python 目標 Python 的起源 Python 解釋器 是用 C 語言實現的,並能夠調用 C 語言的庫文件. Python(蟒蛇) 為什麼要用 Python? Python 的特點 Python 是完全面向對 ...
一周排行
    -Advertisement-
    Play Games
  • 前言 本文介紹一款使用 C# 與 WPF 開發的音頻播放器,其界面簡潔大方,操作體驗流暢。該播放器支持多種音頻格式(如 MP4、WMA、OGG、FLAC 等),並具備標記、實時歌詞顯示等功能。 另外,還支持換膚及多語言(中英文)切換。核心音頻處理採用 FFmpeg 組件,獲得了廣泛認可,目前 Git ...
  • OAuth2.0授權驗證-gitee授權碼模式 本文主要介紹如何筆者自己是如何使用gitee提供的OAuth2.0協議完成授權驗證並登錄到自己的系統,完整模式如圖 1、創建應用 打開gitee個人中心->第三方應用->創建應用 創建應用後在我的應用界面,查看已創建應用的Client ID和Clien ...
  • 解決了這個問題:《winForm下,fastReport.net 從.net framework 升級到.net5遇到的錯誤“Operation is not supported on this platform.”》 本文內容轉載自:https://www.fcnsoft.com/Home/Sho ...
  • 國內文章 WPF 從裸 Win 32 的 WM_Pointer 消息獲取觸摸點繪製筆跡 https://www.cnblogs.com/lindexi/p/18390983 本文將告訴大家如何在 WPF 裡面,接收裸 Win 32 的 WM_Pointer 消息,從消息裡面獲取觸摸點信息,使用觸摸點 ...
  • 前言 給大家推薦一個專為新零售快消行業打造了一套高效的進銷存管理系統。 系統不僅具備強大的庫存管理功能,還集成了高性能的輕量級 POS 解決方案,確保頁面載入速度極快,提供良好的用戶體驗。 項目介紹 Dorisoy.POS 是一款基於 .NET 7 和 Angular 4 開發的新零售快消進銷存管理 ...
  • ABP CLI常用的代碼分享 一、確保環境配置正確 安裝.NET CLI: ABP CLI是基於.NET Core或.NET 5/6/7等更高版本構建的,因此首先需要在你的開發環境中安裝.NET CLI。這可以通過訪問Microsoft官網下載並安裝相應版本的.NET SDK來實現。 安裝ABP ...
  • 問題 問題是這樣的:第三方的webapi,需要先調用登陸介面獲取Cookie,訪問其它介面時攜帶Cookie信息。 但使用HttpClient類調用登陸介面,返回的Headers中沒有找到Cookie信息。 分析 首先,使用Postman測試該登陸介面,正常返回Cookie信息,說明是HttpCli ...
  • 國內文章 關於.NET在中國為什麼工資低的分析 https://www.cnblogs.com/thinkingmore/p/18406244 .NET在中國開發者的薪資偏低,主要因市場需求、技術棧選擇和企業文化等因素所致。歷史上,.NET曾因微軟的閉源策略發展受限,儘管後來推出了跨平臺的.NET ...
  • 在WPF開發應用中,動畫不僅可以引起用戶的註意與興趣,而且還使軟體更加便於使用。前面幾篇文章講解了畫筆(Brush),形狀(Shape),幾何圖形(Geometry),變換(Transform)等相關內容,今天繼續講解動畫相關內容和知識點,僅供學習分享使用,如有不足之處,還請指正。 ...
  • 什麼是委托? 委托可以說是把一個方法代入另一個方法執行,相當於指向函數的指針;事件就相當於保存委托的數組; 1.實例化委托的方式: 方式1:通過new創建實例: public delegate void ShowDelegate(); 或者 public delegate string ShowDe ...