悟空模式-java-工廠方法模式

来源:http://www.cnblogs.com/tirion/archive/2017/09/15/7528293.html
-Advertisement-
Play Games

【卻說那七衣仙女自受了大聖的定身法術,一周天方能解脫,各提花籃,回奏王母說道:“齊天大聖使術法困住我等,故此來遲。”王母問道:“汝等摘了多少蟠桃?”仙女道:“只有兩籃小桃,三籃中桃。至後面,大桃半個也無,想都是大聖偷吃了。及正尋間,不期大聖走將出來,行凶拷打,又問設宴請誰。我等把上會事說了一遍,他就 ...


【卻說那七衣仙女自受了大聖的定身法術,一周天方能解脫,各提花籃,回奏王母說道:“齊天大聖使術法困住我等,故此來遲。”王母問道:“汝等摘了多少蟠桃?”仙女道:“只有兩籃小桃,三籃中桃。至後面,大桃半個也無,想都是大聖偷吃了。及正尋間,不期大聖走將出來,行凶拷打,又問設宴請誰。我等把上會事說了一遍,他就定住我等,不知去向。直到如今,才得醒解回來。”王母聞言,即去見玉帝,備陳前事。】

在《西游記》第五回《亂蟠桃大聖偷丹 反天宮諸神捉怪》里,悟空將奉王母之命前去蟠桃園摘桃的七衣仙女定住,隨即自己偷吃仙桃的事情也因而敗露了。

悟空模式-java-普通工廠模式中,我們舉了王母娘娘(消費者)從蟠桃園(普通工廠類)中獲取蟠桃(產品)的例子來說明普通方法模式,我們發現蟠桃園這個類是普通方法模式的核心類,一旦出現問題,整個設計體系就崩潰了。果不其然,齊天大聖很快就搗亂了,當他把七衣仙女定住的時候,蟠桃園就停止工作,王母娘娘就沒有辦法獲取到蟠桃了,直到過了一周天,蟠桃園恢復運行,王母才拿到幾籃桃子。

那麼接下來我們就嘗試使用工廠方法模式來降低蟠桃系統的風險,主要的改動就是將原本作為核心的蟠桃園類修改為以抽象蟠桃園作為核心,蟠桃的產出交由各個具體的蟠桃園負責,這樣某一個蟠桃園出現問題,不影響其他蟠桃園的正常運行:

蟠桃

package com.tirion.design.simple.factory;

public interface FlatPeach {

    void printLevel();

    void printCycleTime();
}

低級蟠桃

package com.tirion.design.simple.factory;

public class LowLevelFlatPeach implements FlatPeach {

    LowLevelFlatPeach(){
        printCycleTime();
        printLevel();
    }

    @Override
    public void printLevel() {
        System.out.println("低級蟠桃");
    }

    @Override
    public void printCycleTime() {
        System.out.println("三千年一熟");
    }

}

中級蟠桃

package com.tirion.design.simple.factory;

public class MiddleLevelFlatPeach implements FlatPeach {

    MiddleLevelFlatPeach(){
        printCycleTime();
        printLevel();
    }

    @Override
    public void printLevel() {
        System.out.println("中級蟠桃");
    }

    @Override
    public void printCycleTime() {
        System.out.println("六千年一熟");
    }

}

高級蟠桃

package com.tirion.design.simple.factory;

public class HighLevelFlatPeach implements FlatPeach {

    HighLevelFlatPeach(){
        printCycleTime();
        printLevel();
    }

    @Override
    public void printLevel() {
        System.out.println("高級蟠桃");
    }

    @Override
    public void printCycleTime() {
        System.out.println("九千年一熟");
    }

}

抽象蟠桃園

package com.tirion.design.factory.method;

public abstract class FlatPeachGarden {
    public abstract FlatPeach produce();
}

低級蟠桃園

package com.tirion.design.factory.method;

public class LowLevelFlatPeachGarden extends FlatPeachGarden {

    @Override
    public FlatPeach produce() {
        return new LowLevelFlatPeach();
    }
}

中級蟠桃園

package com.tirion.design.factory.method;

public class MiddleLevelFlatPeachGarden extends FlatPeachGarden {

    @Override
    public FlatPeach produce() {
        return new MiddleLevelFlatPeach();
    }
}

高級蟠桃園

package com.tirion.design.factory.method;

public class HighLevelFlatPeachGarden extends FlatPeachGarden {

    @Override
    public FlatPeach produce() {
        return new HighLevelFlatPeach();
    }
}

王母娘娘

package com.tirion.design.factory.method;


public class TheQueenMother {

    public static FlatPeach getLowLevelFlatPeach() {
        FlatPeachGarden garden = new LowLevelFlatPeachGarden();
        FlatPeach flatPeach = garden.produce();
        if (flatPeach == null) {
            System.out.println("王母娘娘沒有得到蟠桃");
        } else {
            System.out.println("王母娘娘獲得了蟠桃");
        }
        return flatPeach;
    }

    public static FlatPeach getMiddleLevelFlatPeach() {
        FlatPeachGarden garden = new MiddleLevelFlatPeachGarden();
        FlatPeach flatPeach = garden.produce();
        if (flatPeach == null) {
            System.out.println("王母娘娘沒有得到蟠桃");
        } else {
            System.out.println("王母娘娘獲得了蟠桃");
        }
        return flatPeach;
    }

    public static FlatPeach getHighLevelFlatPeach() {
        FlatPeachGarden garden = new HighLevelFlatPeachGarden();
        FlatPeach flatPeach = garden.produce();
        if (flatPeach == null) {
            System.out.println("王母娘娘沒有得到蟠桃");
        } else {
            System.out.println("王母娘娘獲得了蟠桃");
        }
        return flatPeach;
    }

    public static void main(String[] args) {
        TheQueenMother.getLowLevelFlatPeach();
        TheQueenMother.getMiddleLevelFlatPeach();
        TheQueenMother.getHighLevelFlatPeach();
    }
}

在上面的代碼中,我們將三種蟠桃分別交給三個不同類別的蟠桃園進行管理,每個蟠桃園負責生產某一種類的蟠桃,這樣就把不同蟠桃的管理拆分開來,不再集中於一個蟠桃園,降低了系統的耦合風險。代碼執行結果如下:

找到對應等級的蟠桃園
三千年一熟
低級蟠桃
王母娘娘獲得了蟠桃
找到對應等級的蟠桃園
六千年一熟
中級蟠桃
王母娘娘獲得了蟠桃
找到對應等級的蟠桃園
九千年一熟
高級蟠桃
王母娘娘獲得了蟠桃

王母娘娘想要哪個種類的蟠桃,就派仙女去對應的蟠桃園去採摘,這就是工廠方法模式的運作方式,上例的類圖如下:

現在我們將工廠方法模式與普通方法模式進行比較,看看工廠方法模式在面對一些問題時會有什麼樣的表現:

現在蟠桃園出現了同樣的情況,悟空把高級蟠桃園裡的桃子全部偷吃光了,又把中間蟠桃園的仙女用定身法術定住了,我們如何處理這樣的變化呢?方法如下:

修改高級蟠桃園類和中級蟠桃園類

package com.tirion.design.factory.method;

public class HighLevelFlatPeachGarden extends FlatPeachGarden {

    @Override
    public FlatPeach produce() {
        System.out.println("高級蟠桃被齊天大聖偷吃光了!");
        return null;
        // return new HighLevelFlatPeach();
    }
}
package com.tirion.design.factory.method;

public class MiddleLevelFlatPeachGarden extends FlatPeachGarden {

    @Override
    public FlatPeach produce() {
        System.out.println("齊天大聖把中級蟠桃園的仙女定住了,沒辦法取蟠桃!");
        return null;
        // return new MiddleLevelFlatPeach();
    }
}

其他類不變,再來看代碼的允許結果:

找到對應等級的蟠桃園
三千年一熟
低級蟠桃
王母娘娘獲得了蟠桃
找到對應等級的蟠桃園
齊天大聖把中級蟠桃園的仙女定住了,沒辦法取蟠桃!
王母娘娘沒有得到蟠桃
找到對應等級的蟠桃園
高級蟠桃被齊天大聖偷吃光了!
王母娘娘沒有得到蟠桃

也就是說,哪個具體的工廠類出現了變化,我們就修改對應的工廠類就可以了,這樣代碼修改不會影響其他沒有發生變化的業務代碼邏輯,符合“單一職責原則”。產品的選擇方案與工廠本身分離開來,工廠只要負責創建產品實例就可以了。

我們再來看另外一種情況,高級蟠桃被悟空吃掉了之後,也許是被齊天大聖的靈氣所激發,部分果核後來居然生長出了一種新的蟠桃樹,一萬年一熟,我們叫它超高級蟠桃。下麵我們將對工廠方法模式的蟠桃園體系進行調整,以適應蟠桃品種的變化:

超高級蟠桃

package com.tirion.design.factory.method;

public class SuperHighLevelFlatPeach implements FlatPeach {

    SuperHighLevelFlatPeach(){
        printCycleTime();
        printLevel();
    }

    @Override
    public void printLevel() {
        System.out.println("超高級蟠桃");
    }

    @Override
    public void printCycleTime() {
        System.out.println("一萬年一熟");
    }

}

超高級蟠桃園

package com.tirion.design.factory.method;

public class SuperHighLevelFlatPeachGarden extends FlatPeachGarden {

    @Override
    public FlatPeach produce() {
        return new SuperHighLevelFlatPeach();
    }
}

這樣,我們就把這種新的蟠桃品種加入進來了。王母娘娘暫時還不知道有了新的蟠桃,哪天她知道了,就派人去超高級蟠桃園去採摘就可以了。

從上面可以看出,增加一個新的蟠桃品種只需要增加一個新的產品類(超高級蟠桃)和一個新的產品工廠(超高級蟠桃園),而不必更改原有體系中任何一個類的代碼,這樣就符合了“開閉原則”。

所以說,工廠方法模式是普通工廠模式為了適應更多的未來變化而衍生出來的更高級的工廠模式,它解決了一些普通工廠模式違背的設計原則問題,即單一職責原則開閉原則

但是,工廠方法模式也有它的缺陷,那就是隨著產品越來越多,每個產品都需要新建一個產品類和對應的工廠類,這樣最終會導致類非常多。消費者也很難管理這些類。以上面的例子來說,就是最終蟠桃園裡有好幾十種蟠桃,王母娘娘也搞不清楚到底要哪些蟠桃,派仙女去哪些蟠桃園去採摘,最終引發體系混亂,代碼修改難度加大,風險也隨之上升。

為瞭解決這個問題,我們引入了悟空模式-java-抽象工廠模式,關於工廠方法模式的介紹就到這裡,你可以將它記憶為多蟠桃園模式

如果你認為文章中哪裡有錯誤或者不足的地方,歡迎在評論區指出,也希望這篇文章對你學習java設計模式能夠有所幫助。轉載請註明,謝謝。


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

-Advertisement-
Play Games
更多相關文章
  • PHP+MySql+Ajax實現年月日的三級聯動 <!DOCTYPE html><html> <head> <meta charset="UTF-8"> <title></title> <script src="bootstrap/js/jquery-1.11.2.min.js"></script> ...
  • 這裡說下最近項目中我們的解決方案,主要用到commons-lang3-3.1.jar這個包的org.apache.commons.lang3.StringEscapeUtils.escapeHtml4()這個方法。 解決過程主要在用戶輸入和顯示輸出兩步:在輸入時對特殊字元如<>" ' & 轉義,在輸 ...
  • 前幾天實現了vmmap類似的功能,就是每次記憶體分配釋放,都將其調用堆棧信息保存起來 已經生成了兩個文件,alloc.mapping和address.mapping,剩下的就是編寫工具解析這兩個文件了. 後續要做的:1.無需編譯,直接dll註入,開啟單獨線程寫上述兩個文件;2.編寫gui工具,指定時間... ...
  • 插入排序 の implementation 插入排序就像打賭的時候,比如雙扣。抽牌的時候,一次拿一張牌,這張牌和之前的牌一張張比較。選擇把這張牌插入什麼位置,排好順序的位置後打牌更順。要不然得一個一個找到時麻煩。也不利於打牌的大局觀。看下圖 假設第一次抽到梅花7, 不用排序。因為只有一張 接著抽到梅 ...
  • 溫度PID控制時, 有多個溫度測量點, 但最終只有一個溫度作為反饋值參與PID控制, 我們有時會採用平均值法, 計算所有溫度的平均值來作為PID的反饋值. 但在某一路溫度探頭斷線的情況下, 溫度值為零, 這時平均值就會小很多.像這種情況下中值法比較有效. 原理如下: 對於一組溫度, 模擬量轉換為數字... ...
  • 編譯:Lemon,原文作者:Ernst Haagsman 公眾號:Python數據之道(ID:PyDataRoad) pycharm:2017.3版即將新增科學計算模式 在JetBrains將發佈的新版本PyCharm中,考慮到越來越多的Python開發者進行數據科學相關開發,即將發佈的PyChar ...
  • 首先,看完這篇文章,不能保證你成為大神,但是卻可以讓你懂得什麼是爬蟲,如何使用爬蟲,如何利用http協議,侵入別人的系統,當然只是一些簡單的教程,拿到一些簡單的數據; 先上代碼,在一步一步講解: 這是一個工具類,不用詳細看,網上哪裡都能找到發送http請求的工具類,少包自己導 package com ...
  • 筆者每當遇到一個新事物的想去瞭解的時候,總是會問上自己第一個問題,這個新事物是乾什麼用的?在解釋我所理解的REST這個過程中也不例外,這篇博客我們先關註一下REST是乾什麼用的,然後後續再解釋REST是什麼。好了,不說廢話了,在上篇結尾處說到,REST是Web的架構風格,暫且不管REST和Web,我 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...