悟空模式-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
  • 前言 本文介紹一款使用 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 ...