單例模式: 即在整個生命周期中,對於該對象的生產始終都是一個,不曾變化。 保證了一個類僅有一個實例,並提供一個訪問它的全局訪問點。 作用: 在要求線程安全的情況下,保證了類實例的唯一性,線程安全。 在不需要多實例存在時,保證了類實例的單一性。不浪費記憶體。 特點: 公有的方法獲取實例, 私有的構造方法 ...
單例模式:
- 即在整個生命周期中,對於該對象的生產始終都是一個,不曾變化。
- 保證了一個類僅有一個實例,並提供一個訪問它的全局訪問點。
作用:
- 在要求線程安全的情況下,保證了類實例的唯一性,線程安全。
- 在不需要多實例存在時,保證了類實例的單一性。不浪費記憶體。
特點:
- 公有的方法獲取實例,
- 私有的構造方法,
- 私有的成員變數。
一,餓漢式
* @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併發編程詳解》讀書筆記