單例模式作用特點及常見的單例模式分析(6種單例模式分析)


單例模式:

  1. 即在整個生命周期中,對於該對象的生產始終都是一個,不曾變化。
  2. 保證了一個類僅有一個實例,並提供一個訪問它的全局訪問點。

作用:

  1. 在要求線程安全的情況下,保證了類實例的唯一性,線程安全。
  2. 在不需要多實例存在時,保證了類實例的單一性。不浪費內存。

特點:

  1. 公有的方法獲取實例,
  2. 私有的構造方法,
  3. 私有的成員變量。

 

一,餓漢式
* @Description 餓漢式單例
     * 餓漢式單例關鍵在於singleton作為類變量並且直接得到了初始化,即類中所有的變量都會被初始化
     * singleton作為類變量在初始化的過程中會被收集進<clinit>()方法中,該方法能夠百分之百的保證同步,
     * 但是因為不是懶加載,singleton被加載后可能很長一段時間不被使用,即實例所開辟的空間會存在很長時間
     * 雖然可以實現多線程的唯一實例,但無法進行懶加載;

package com.liruilong.singleton;
 
/**
 * @Author: Liruilong
 * @Date: 2019/7/20 17:55
 */
 
// final 不允許被繼承
public final class Singleton {
    // 實例變量
    private byte[] bate = new byte[1024];
    // 私有的構造函數,即不允許外部 new
    private Singleton(){ }
    private  static final Singleton singleton1 = new Singleton();
    public static  Singleton getInstance1(){
        return singleton1;
    }

二,懶漢式

* @Description 懶漢式單例模式
     * 可以保證懶加載,但是線程不安全
     * 當有兩個線程訪問時,不能保證單例的唯一性

package com.liruilong.singleton;
 
/**
 * @Author: Liruilong
 * @Date: 2019/7/20 17:55
 */
 
// final 不允許被繼承
public final class Singleton {
    // 實例變量
    private byte[] bate = new byte[1024];
    // 私有的構造函數,即不允許外部 new
    private Singleton(){ }
 
    private static  Singleton singleton =null;
public static Singleton getInstance(){ if (singleton == null) { singleton = new Singleton(); } return singleton; }

三,懶漢式加同步方法
* @Description 懶漢式+同步方法單例模式
     * 即能保證懶加載,又可以保證singleton實例的唯一性,但是synchronizeed關鍵字的排他性導致
     * getInstance0()方法只能在同一時間被一個線程訪問。性能低下。

 

package com.liruilong.singleton;
 
/**
 * @Author: Liruilong
 * @Date: 2019/7/20 17:55
 */
 
// final 不允許被繼承
public final class Singleton {
    // 實例變量
    private byte[] bate = new byte[1024];
    // 私有的構造函數,即不允許外部 new
    private Singleton(){ }

    private static  Singleton singleton =null;
public static synchronized Singleton getInstance0(){ if (singleton == null) { singleton = new Singleton(); } return singleton; }

四,雙重效驗鎖單例
* @Description 雙重校驗鎖單例(Double-Check)+Volatile
     *  對懶漢-同步方法的改進,當有兩個線程發現singleton為null時,只有一個線程可以進入到同步代碼塊里。
     *  即滿足了懶加載,又保證了線程的唯一性
     *  不加volition的缺點,有時候可能會報NPE,(JVM運行指令重排序)
     *  有可能實例對象的變量未完成實例化其他線程去獲取到singleton變量。
     *  未完成初始化的實例調用其方法會拋出空指針異常。

 

package com.liruilong.singleton;
 
/**
 * @Author: Liruilong
 * @Date: 2019/7/20 17:55
 */
 
// final 不允許被繼承
public final class Singleton {
    // 實例變量
    private byte[] bate = new byte[1024];
    // 私有的構造函數,即不允許外部 new
    private Singleton(){ }
 
    private  static volatile Singleton singleton2 = null;
public static Singleton getInstance4() { if (singleton2 == null){ synchronized (Singleton.class){ if (singleton2 ==null){ singleton2 = new Singleton(); } } } return singleton2; }

 

五,靜態內部類單例
 * @Description 靜態內部類的單例模式
     * 在Singleton類初始化並不會創建Singleton實例,在靜態內部類中定義了singleton實例。
     * 當給靜態內部類被主動創建時則會創建Singleton靜態變量,是最好的單例模式之一
   

package com.liruilong.singleton;
 
/**
 * @Author: Liruilong
 * @Date: 2019/7/20 17:55
 */
 
// final 不允許被繼承
public final class Singleton {
    // 實例變量
    private byte[] bate = new byte[1024];
    // 私有的構造函數,即不允許外部 new
    private Singleton(){ }
 
    private  static class Singtetons{
private static Singleton SINGLETON = new Singleton(); /* static { final Singleton SINGLETON = new Singleton(); }*/ } public static Singleton getInstance2(){ return Singtetons.SINGLETON; }

六,枚舉類單例 

 * @Description 基於枚舉類線程安全
     * 枚舉類型不允許被繼承,同樣線程安全的,且只能被實例化一次。

 

package com.liruilong.singleton;
 
/**
 * @Author: Liruilong
 * @Date: 2019/7/20 17:55
 */
 
// final 不允許被繼承
public final class Singleton {
    // 實例變量
    private byte[] bate = new byte[1024];
    // 私有的構造函數,即不允許外部 new
    private Singleton(){ }

    private enum Singtetonss {
        SINGTETONSS; //實例必須第一行,默認 public final static修飾
        private Singleton singleton;
 
        Singtetonss() { //構造器。默認私有
            this.singleton = new Singleton();
        }
        public static Singleton getInstance() {
            return SINGTETONSS.singleton;
        }
    }
    public static  Singleton getInstance3(){
        return Singtetonss.getInstance();
    }

 

原本是我筆記里的,摘了出來,面試的時候看,

更多見 ----》《Java並發編程詳解》讀書筆記

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM