您好,登錄后才能下訂單哦!
1 餓漢式
public class Singleton1 { //不能延遲加載 占用內(nèi)存 耗費(fèi)資源 private static Singleton1 singleton1 = new Singleton1(); public static Singleton1 getSingleton1() { return singleton1; } }
可以保證多個(gè)線程下唯一實(shí)例,getSingleton1 方法性能較高,但是無(wú)法進(jìn)行懶加載。
2 懶漢式
public class Singleton2 { //延遲加載 // 多線程下 不安全 private static Singleton2 singleton1 = null; public Singleton2 getSingleton1() { if (singleton1==null){ singleton1 = new Singleton2(); } return singleton1; } }
懶漢式 解決了延遲加載和資源問(wèn)題,但是多線程下存在線程不安全問(wèn)題。
3 懶漢式 + 同步
public class Singleton3 { //延遲加載 // 多線程下 不安全 private static Singleton3 singleton1 = null; //解決延遲加載 多線程安全問(wèn)題,但存在讀操作,加鎖問(wèn)題,線程排隊(duì),寫(xiě)操作只有一次 獲取時(shí)需要排隊(duì)等候問(wèn)題 public synchronized Singleton3 getSingleton1() { if (singleton1==null){ singleton1 = new Singleton3(); } return singleton1; } /* 等同方法前加鎖 public static Singleton3 getSingleton1() { synchronized(Singleton3.class){ if (singleton1==null){ singleton1 = new Singleton3(); } } return singleton1; } */ }
解決延遲加載 多線程安全問(wèn)題,但存在讀操作,加鎖問(wèn)題,線程排隊(duì),寫(xiě)操作(創(chuàng)建對(duì)象)只有一次 ,但是獲取時(shí)需要排隊(duì)等候問(wèn)題
4 懶漢式 + 雙重檢驗(yàn)
public class Singleton4 { //延遲加載 private static Singleton4 singleton1 = null; // 解決 讀操作 多線程情況下 排隊(duì)獲取問(wèn)題, 但是雙重校驗(yàn) 也存在一個(gè)問(wèn)題,jvm 重排序的問(wèn)題下 會(huì)存在空指針問(wèn)題 public static Singleton4 getSingleton1() { if (singleton1==null){ synchronized (Singleton4.class) { if (singleton1 == null) { singleton1 = new Singleton4(); } } } return singleton1; } }
解決 讀操作 多線程情況下 排隊(duì)獲取問(wèn)題, 但是雙重校驗(yàn) 也存在一個(gè)問(wèn)題,jvm 重排序的問(wèn)題下 會(huì)存在空指針問(wèn)題
但存在一個(gè)問(wèn)題,jvm指令重排序, JVM 的即時(shí)編譯器中存在指令重排序的優(yōu)化。
1 首先給 singleton1 分配內(nèi)存
2 Singleton4 執(zhí)行構(gòu)造函數(shù) 開(kāi)辟空間
3 調(diào)用getSingleton1()方法創(chuàng)建對(duì)象
JVM 的即時(shí)編譯器中存在指令重排序的優(yōu)化
理想情況下 jvm執(zhí)行順序是123 也可能是 132 ,13在創(chuàng)建完對(duì)象后 ,再執(zhí)行2 返回null,此時(shí)就是空指針了。
5 懶漢式 + 雙重檢驗(yàn) + volatile
volatile 關(guān)鍵字 禁止JVM編譯時(shí)指令重排序
public class Singleton5 { //延遲加載 // volatile 關(guān)鍵字 禁止指令重排序 // 解決 雙重校驗(yàn) 也存在一個(gè)問(wèn)題,jvm 重排序的問(wèn)題下 會(huì)存在空指針問(wèn)題 private static volatile Singleton5 singleton1 = null; public static Singleton5 getSingleton1() { if (singleton1==null){ synchronized (Singleton5.class) { if (singleton1 == null) { singleton1 = new Singleton5(); } } } return singleton1; } }
6 靜態(tài)內(nèi)部類(lèi)
public class Singleton6 { //延遲加載 //靜態(tài)內(nèi)部類(lèi) 靜態(tài)的始終在jvm中存在一份 static class Singleton { private static Singleton6 singleton1 = new Singleton6(); } public static Singleton6 get(){ return Singleton.singleton1; } }
7 枚舉
public class Singleton7 { //枚舉類(lèi)型是 線程安全 構(gòu)造方法只會(huì)被裝載一次 private enum Singleton { Singleton; private final Singleton7 singleton7; Singleton() { singleton7 = new Singleton7(); } public Singleton7 getSingleton7() { return singleton7; } } //延遲加載 public static Singleton7 get() { return Singleton.Singleton.getSingleton7(); } //測(cè)試 public static void main(String[] args) { IntStream.rangeClosed(1, 100).forEach(i -> { new Thread(String.valueOf(i)) { @Override public void run() { System.out.println(Singleton7.get()); } }.start(); }); } }
枚舉類(lèi)型不允許被繼承,但線程是安全的,只能被實(shí)例化一次,但是枚舉類(lèi)型不能夠懶加載,和方法配合使用,調(diào)用get()靜態(tài)方法,然后singleton7會(huì)延遲加載得到實(shí)例化。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持億速云。
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。