詳解JAVA 七種創(chuàng)建單例的方法
1 餓漢式
public class Singleton1 {
//不能延遲加載 占用內存 耗費資源
private static Singleton1 singleton1 = new Singleton1();
public static Singleton1 getSingleton1() {
return singleton1;
}
}
可以保證多個線程下唯一實例,getSingleton1 方法性能較高,但是無法進行懶加載。
2 懶漢式
public class Singleton2 {
//延遲加載
// 多線程下 不安全
private static Singleton2 singleton1 = null;
public Singleton2 getSingleton1() {
if (singleton1==null){
singleton1 = new Singleton2();
}
return singleton1;
}
}
懶漢式 解決了延遲加載和資源問題,但是多線程下存在線程不安全問題。
3 懶漢式 + 同步
public class Singleton3 {
//延遲加載
// 多線程下 不安全
private static Singleton3 singleton1 = null;
//解決延遲加載 多線程安全問題,但存在讀操作,加鎖問題,線程排隊,寫操作只有一次 獲取時需要排隊等候問題
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;
}
*/
}
解決延遲加載 多線程安全問題,但存在讀操作,加鎖問題,線程排隊,寫操作(創(chuàng)建對象)只有一次 ,但是獲取時需要排隊等候問題
4 懶漢式 + 雙重檢驗
public class Singleton4 {
//延遲加載
private static Singleton4 singleton1 = null;
// 解決 讀操作 多線程情況下 排隊獲取問題, 但是雙重校驗 也存在一個問題,jvm 重排序的問題下 會存在空指針問題
public static Singleton4 getSingleton1() {
if (singleton1==null){
synchronized (Singleton4.class) {
if (singleton1 == null) {
singleton1 = new Singleton4();
}
}
}
return singleton1;
}
}
解決 讀操作 多線程情況下 排隊獲取問題, 但是雙重校驗 也存在一個問題,jvm 重排序的問題下 會存在空指針問題
但存在一個問題,jvm指令重排序, JVM 的即時編譯器中存在指令重排序的優(yōu)化。
1 首先給 singleton1 分配內存
2 Singleton4 執(zhí)行構造函數(shù) 開辟空間
3 調用getSingleton1()方法創(chuàng)建對象
JVM 的即時編譯器中存在指令重排序的優(yōu)化
理想情況下 jvm執(zhí)行順序是123 也可能是 132 ,13在創(chuàng)建完對象后 ,再執(zhí)行2 返回null,此時就是空指針了。
5 懶漢式 + 雙重檢驗 + volatile
volatile 關鍵字 禁止JVM編譯時指令重排序
public class Singleton5 {
//延遲加載
// volatile 關鍵字 禁止指令重排序
// 解決 雙重校驗 也存在一個問題,jvm 重排序的問題下 會存在空指針問題
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)內部類
public class Singleton6 {
//延遲加載
//靜態(tài)內部類 靜態(tài)的始終在jvm中存在一份
static class Singleton {
private static Singleton6 singleton1 = new Singleton6();
}
public static Singleton6 get(){
return Singleton.singleton1;
}
}
7 枚舉
public class Singleton7 {
//枚舉類型是 線程安全 構造方法只會被裝載一次
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();
}
//測試
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();
});
}
}
枚舉類型不允許被繼承,但線程是安全的,只能被實例化一次,但是枚舉類型不能夠懶加載,和方法配合使用,調用get()靜態(tài)方法,然后singleton7會延遲加載得到實例化。
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。
相關文章
SpringCloud集成MybatisPlus實現(xiàn)MySQL多數(shù)據(jù)源配置方法
本文詳細介紹了SpringCloud集成MybatisPlus實現(xiàn)MySQL多數(shù)據(jù)源配置的方法,包括在application.properties中配置多數(shù)據(jù)源,配置MybatisPlus,創(chuàng)建Mapper接口和使用多數(shù)據(jù)源等步驟,此外,還解釋了每一個配置項目的含義,以便讀者更好地理解和應用2024-10-10
SpringBoot application.yml和bootstrap.yml的區(qū)別
本文主要介紹了SpringBoot application.yml和bootstrap.yml的區(qū)別,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2023-04-04
SpringBoot重寫addResourceHandlers映射文件路徑方式
這篇文章主要介紹了SpringBoot重寫addResourceHandlers映射文件路徑方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-02-02
詳解mybatis-plus使用@EnumValue注解的方式對枚舉類型的處理
這篇文章主要介紹了詳解mybatis-plus使用@EnumValue注解的方式對枚舉類型的處理,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2020-12-12
SpringMVC基于阻塞隊列LinkedBlockingQueue的同步長輪詢功能實現(xiàn)詳解
這篇文章主要介紹了SpringMVC基于阻塞隊列LinkedBlockingQueue的同步長輪詢功能實現(xiàn)詳解,本文介紹的也是生產(chǎn)者消費者的一種實現(xiàn),生產(chǎn)者不必是一個始終在執(zhí)行的線程,它可以是一個接口,接受客戶端的請求,向隊列中插入消息,需要的朋友可以參考下2023-07-07

