Android的設計模式-單例模式

前言

Android的設計模式系列文章介紹:歡迎關注,持續更新中:

Android的設計模式-設計模式的六大原則
一句話總結23種設計模式則
創建型模式:
Android的設計模式-單例模式
Android的設計模式-建造者模式
Android的設計模式-工廠方法模式
Android的設計模式-簡單工廠模式
Android的設計模式-抽象工廠模式
Android的設計模式-原型模式
行為型模式:
Android的設計模式-策略模式
Android的設計模式-狀態模式
Android的設計模式-責任鏈模式
Android的設計模式-觀察者模式
Android的設計模式-模板方法模式
Android的設計模式-迭代器模式
Android的設計模式-備忘錄模式
Android的設計模式-訪問者模式
Android的設計模式-中介者模式
Android的設計模式-解釋器模式
Android的設計模式-命令模式
結構型模式:
Android的設計模式-代理模式
Android的設計模式-組合模式
Android的設計模式-適配器模式
Android的設計模式-裝飾者模式
Android的設計模式-享元模式
Android的設計模式-外觀模式
Android的設計模式-橋接模式

1.定義

確保某個類只有一個實例,并且自行實例化并向整個系統提供這個實例。

2.介紹

單例模式屬于創建類模式。
單例模式有以下特點:

 1. 單例類只能有一個實例。
 2. 單例類必須自己創建自己的唯一實例。
 3. 單例類必須給所有其他對象提供這一實例。

3.實現

3.4 餓漢式
//單例類.   
public class Singleton {
    
    private Singleton() {//構造方法為private,防止外部代碼直接通過new來構造多個對象
    }

    private static final Singleton single = new Singleton();  //在類初始化時,已經自行實例化,所以是線程安全的。

    public static Singleton getInstance() {  //通過getInstance()方法獲取實例對象
        return single;
    }
}  
  • 優點:寫法簡單,線程安全。
  • 缺點:沒有懶加載的效果,如果沒有使用過的話會造成內存浪費。
3.1 懶漢式(線程不安全)
//單例類
public class Singleton {
    private Singleton() {
    }

    private static Singleton single = null;

    public static Singleton getInstance() {
        if (single == null) {
            single = new Singleton();  //在第一次調用getInstance()時才實例化,實現懶加載,所以叫懶漢式
        }
        return single;
    }
} 
  • 優點:實現了懶加載的效果。
  • 缺點:線程不安全。
3.1 懶漢式(線程安全)
//單例類
public class Singleton {
    private Singleton() {
    }

    private static Singleton single = null;

    public static synchronized Singleton getInstance() { //加上synchronized同步 
        if (single == null) {
            single = new Singleton();
        }
        return single;
    }
}  
  • 優點:實現了懶加載的效果,線程安全。
  • 缺點:使用synchronized會造成不必要的同步開銷,而且大部分時候我們是用不到同步的。
3.3 雙重檢查鎖定(DCL)
public class Singleton {
    private volatile static Singleton singleton; //volatile 能夠防止代碼的重排序,保證得到的對象是初始化過

    private Singleton() {
    }

    public static Singleton getSingleton() {
        if (singleton == null) {  //第一次檢查,避免不必要的同步
            synchronized (Singleton.class) {  //同步
                if (singleton == null) {   //第二次檢查,為null時才創建實例
                    singleton = new Singleton();
                }
            }
        }
        return singleton;
    }
} 
  • 優點:懶加載,線程安全,效率較高
  • 缺點:volatile影響一點性能,高并發下有一定的缺陷,某些情況下DCL會失效,雖然概率較小。
3.5 靜態內部類
public class Singleton {
    private Singleton() {
    }

    public static Singleton getInstance() {
        //第一次調用getInstance方法時才加載SingletonHolder并初始化sInstance
        return SingletonHolder.sInstance;
    }
    
    //靜態內部類
    private static class SingletonHolder {
        private static final Singleton sInstance = new Singleton();
    }
}
  • 優點:懶加載,線程安全,推薦使用
3.6 枚舉單例
public enum Singleton {

    INSTANCE;   //定義一個枚舉的元素,它就是Singleton的一個實例

    public void doSomething() {
    }
}  
  • 優點:線程安全,寫法簡單,能防止反序列化重新創建新的對象。
  • 缺點:可讀性不高,枚舉會比靜態常量多那么一丁點的內存。
3.7 使用容器實現單例模式
//單例管理類
public class SingletonManager {
    private static Map<String, Object> objMap = new HashMap<String, Object>();

    public static void registerService(String key, Object instance) {
        if (!objMap.containsKey(key)) {
            objMap.put(key, instance);//添加單例
        }
    }

    public static Object getService(String key) {
        return objMap.get(key);//獲取單例
    }
}
  • 優點:方便管理。
  • 缺點:寫法稍復雜。

4.注意事項

  1. 使用反射能夠破壞單例模式,所以應該慎用反射
    Constructor con = Singleton.class.getDeclaredConstructor();
    con.setAccessible(true);
    // 通過反射獲取實例
    Singleton singeton1 = (Singleton) con.newInstance();
    Singleton singeton2 = (Singleton) con.newInstance();
    System.out.println(singeton1==singeton2);//結果為false,singeton1和singeton2將是兩個不同的實例
  • 可以通過當第二次調用構造函數時拋出異常來防止反射破壞單例,以懶漢式為例:
public class Singleton {
    private static boolean flag = true;
    private static Singleton single = null;

    private Singleton() {
        if (flag) {
            flag = !flag;
        } else {
            throw new RuntimeException("單例模式被破壞!");
        }
    }

    public static Singleton getInstance() {
        if (single == null) {
            single = new Singleton();
        }
        return single;
    }
}  
  1. 反序列化時也會破壞單例模式,可以通過重寫readResolve方法避免,以餓漢式為例:
public class Singleton implements Serializable {
    private Singleton() {
    }

    private static final Singleton single = new Singleton();

    public static Singleton getInstance() {
        return single;
    }

    private Object readResolve() throws ObjectStreamException {//重寫readResolve()
        return single;//直接返回單例對象
    }
} 

5.應用場景

  • 頻繁訪問數據庫或文件的對象。
  • 工具類對象;
  • 創建對象時耗時過多或耗費資源過多,但又經常用到的對象;

6.優點

  • 內存中只存在一個對象,節省了系統資源。
  • 避免對資源的多重占用,例如一個文件操作,由于只有一個實例存在內存中,避免對同一資源文件的同時操作。

7.缺點

  • 獲取對象時不能用new
  • 單例對象如果持有Context,那么很容易引發內存泄露。
  • 單例模式一般沒有接口,擴展很困難,若要擴展,只能修改代碼來實現。

相關文章閱讀
Android的設計模式-設計模式的六大原則
一句話總結23種設計模式則
創建型模式:
Android的設計模式-單例模式
Android的設計模式-建造者模式
Android的設計模式-工廠方法模式
Android的設計模式-簡單工廠模式
Android的設計模式-抽象工廠模式
Android的設計模式-原型模式
行為型模式:
Android的設計模式-策略模式
Android的設計模式-狀態模式
Android的設計模式-責任鏈模式
Android的設計模式-觀察者模式
Android的設計模式-模板方法模式
Android的設計模式-迭代器模式
Android的設計模式-備忘錄模式
Android的設計模式-訪問者模式
Android的設計模式-中介者模式
Android的設計模式-解釋器模式
Android的設計模式-命令模式
結構型模式:
Android的設計模式-代理模式
Android的設計模式-組合模式
Android的設計模式-適配器模式
Android的設計模式-裝飾者模式
Android的設計模式-享元模式
Android的設計模式-外觀模式
Android的設計模式-橋接模式

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 227,882評論 6 531
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,208評論 3 414
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 175,746評論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,666評論 1 309
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,477評論 6 407
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 54,960評論 1 321
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,047評論 3 440
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,200評論 0 288
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,726評論 1 333
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,617評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,807評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,327評論 5 358
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,049評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,425評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,674評論 1 281
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,432評論 3 390
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,769評論 2 372

推薦閱讀更多精彩內容

  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,610評論 25 708
  • 單例模式(SingletonPattern)一般被認為是最簡單、最易理解的設計模式,也因為它的簡潔易懂,是項目中最...
    成熱了閱讀 4,279評論 4 34
  • 1.單例模式概述 (1)引言 單例模式是應用最廣的模式之一,也是23種設計模式中最基本的一個。本文旨在總結通過Ja...
    曹豐斌閱讀 2,967評論 6 47
  • 我們經常遇到vue中v-for一個列表,列表的每一項都綁定了@click處理事件。我們都知道綁定這么多監聽,從性能...
    overflow_hidden閱讀 18,308評論 0 3
  • 春雨拂面楊柳風,桃謝櫻開曲觴中。 醉臥錢塘君莫笑,一片碧葉寄浮生。
    慕容錦瑟閱讀 186評論 0 2