淺析Java設(shè)計模式編程中的單例模式和簡單工廠模式
單例模式
動機
有時候只有一個類的實例是很重要的。比如,一個系統(tǒng)應(yīng)該只有一個窗口管理實例。
單例模式是最簡單設(shè)計模式:類負責實例化自己,確保只有一個實例,并且提供一個訪問這個實例的入口。
目的
1. 確保只有一個實例被創(chuàng)建。
2. 提供訪問這個實例的入口。
使用final確保被創(chuàng)建一次,private的構(gòu)造函數(shù)確保不被實例化。public的getInstance方法確保外部能夠訪問。下面是餓漢模式:
public class Singleton { private static final Singleton instance = new Singleton(); private Singleton() {} public static Singleton getInstance() { return instance; } }
懶漢模式:
public class SingletonDemo { private static volatile SingletonDemo instance = null; private SingletonDemo() { } public static SingletonDemo getInstance() { if (instance == null) { synchronized (SingletonDemo .class){ if (instance == null) { instance = new SingletonDemo (); } } } return instance; } }
適用場景和實例
1. Logger類,防止每次打印log的使用都創(chuàng)建一個Logger實例。
2. 控制類,一般整個系統(tǒng)都只有一個控制實例。
具體問題和實現(xiàn)
1. 線程安全,健壯的單例模式應(yīng)該是線程安全的。
2. 懶漢模式使用了雙重鎖機制。
3. 餓漢模式使用靜態(tài)變量,在程序加載時就實例化,保證了只有一個實例。
4. 抽象工廠和工廠方法通常被設(shè)計成單例模式,以保證只有一個工廠。
5. 使用序列化和反序列化時,會有多個實例被創(chuàng)建,使用readResolve函數(shù)避免這個情況,不過最好是不要使用序列化。
public class Singleton implements Serializable { ... // This method is called immediately after an object of this class is deserialized. // This method returns the singleton instance. protected Object readResolve() { return getInstance(); } }
關(guān)鍵點
1. 在多線程的程序中,要注意數(shù)據(jù)的同步。
2. 序列化時要使用readResolve方法返回實例,避免多個對象被創(chuàng)建。
3. 如果被多個類加載器加載時,會有多個實例被創(chuàng)建。
簡單工廠模式
動機
簡單工廠模式是抽象工廠和工廠方法的基礎(chǔ)和初步實現(xiàn)。
目的
1. 不向客戶透露對象實例化的細節(jié)。
2. 通過通用接口創(chuàng)建對象。
實現(xiàn)
實現(xiàn)非常簡單:
1. Client需要Product時,不使用new來創(chuàng)建,而是提供 Product 描述給Factory,讓 Factory 提供一個新的 Product 。
2. Factory實例化一個Product給Client。
3. Client使用抽象Product,而不關(guān)心Product的具體實現(xiàn)。
實例
1. 繪制形狀的繪圖程序。形狀就是Product接口,三角形這些是Concrete Product,我們可以創(chuàng)建一個工廠,然后根據(jù)客戶的描述創(chuàng)建對于的產(chǎn)品。不過添加新的形狀時,我們需要修改工廠類。
具體問題和實現(xiàn)
1. 添加新產(chǎn)品時,需要修改工廠。
public class ProductFactory{ public Product createProduct(String ProductID){ if (id==ID1) return new OneProduct(); if (id==ID2) return new AnotherProduct(); ... // so on for the other Ids return null; //if the id doesn't have any of the expected values } ... }
一般我們通過if語句判斷產(chǎn)品描述,并實例化不同的產(chǎn)品,有新的產(chǎn)品時,我們需要增加新的判斷。通過抽象工廠模式可以解決這個問題。
總結(jié)
1. 當你確實需要工廠模式時才使用,不然只是增加程序的復(fù)雜度,比如多種對象有相似的基本類型時,可以考慮使用簡單工廠模式來統(tǒng)一創(chuàng)建對象。
2. 簡單工廠有比較多的判斷分支語句,違反了開閉原則的對修改關(guān)閉的原則,所以,明智的做法是,對一些固定和簡單程序使用簡單工廠模式,對一些復(fù)雜和需要經(jīng)常擴展的程序,使用抽象工廠模式或者工廠方法模式。
相關(guān)文章
Java 實現(xiàn)倒計時功能(由秒計算天、小時、分鐘、秒)
最近做項目遇到這樣的需求,天、小時、分鐘、秒的數(shù)值都是隔開的,服務(wù)器端只返回一個時間戳長度,怎么實現(xiàn)這樣的功能呢?下面小編給大家?guī)砹薐ava 實現(xiàn)倒計時功能的方案,需要的朋友參考下吧2018-01-01解決mapper.xml中resultType映射類型的問題
這篇文章主要介紹了解決mapper.xml中resultType映射類型的問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2023-06-06解析springBoot-actuator項目構(gòu)造中health端點工作原理
這篇文章主要介紹了springBoot-actuator中health端點工作原理,對spring-boot-actuator的項目構(gòu)造,工作原理進行了全面的梳理,側(cè)重health健康檢查部分2022-02-02SpringBoot實現(xiàn)ORM操作MySQL的幾種方法
本文主要介紹了SpringBoot實現(xiàn)ORM操作MySQL的幾種方法,文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2022-02-02SpringCloud Eureka 服務(wù)注冊實現(xiàn)過程
這篇文章主要介紹了SpringCloud Eureka 服務(wù)注冊實現(xiàn)過程,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2019-10-10