Java 如何安全的發佈對象
安全發佈對象
- 在靜態初始化函數中初始化一個對象引用
- 將對象的引用保存到volatile類型域或者AtomicReference對象中
- 將對象的引用保存到某個正確構造對象的final類型域中
- 將對象的引用保存到一個由鎖保護的域中
Spring 框架中,Spring管理的類都是單例模式。如何保證一個實例隻被初始化一次,且線程安全?通過不同單例的寫法,具體描述安全發佈對象的四種方法:
在靜態初始化函數中初始化一個對象的引用(不推薦)
package com.rumenz.task.single; //線程安全 //餓漢模式 //靜態代碼塊初始化 public class SingletonExample { private SingletonExample(){ //初始化操作 } private static SingletonExample singletonExample=null; static { singletonExample=new SingletonExample(); } public static SingletonExample getInstance(){ return singletonExample; } } //或者 package com.rumenz.task.single; //線程安全 //餓漢模式 //靜態代碼塊初始化 public class SingletonExample { private SingletonExample(){ //初始化操作 } private static SingletonExample singletonExample=new SingletonExample(); public static SingletonExample getInstance(){ return singletonExample; } }
缺點:用不用都會初始化對象,如果初始化工作較多,加載速度會變慢,影響系統性能。
將對象的引用保存到volatile類型或AtomicReference對象中(推薦)
package com.rumenz.task.single; //線程安全 //懶漢模式 public class SingletonExample1 { private SingletonExample1() { //初始化操作 } // 1、memory = allocate() 分配對象的內存空間 // 2、ctorInstance() 初始化對象 // 3、instance = memory 設置instance指向剛分配的內存 // 單例對象 volatile + 雙重檢測機制 -> 禁止指令重排 private volatile static SingletonExample1 singletonExample1=null; //靜態工廠方法 public static SingletonExample1 getInstance(){ if(singletonExample1==null){ //雙重檢測 synchronized(SingletonExample1.class){ //同步鎖 if(singletonExample1==null){ singletonExample1=new SingletonExample1(); } } } return singletonExample1; } }
優點:按需加載
缺點:第一次初始化的時候可能會比較慢
通過synchronized(不推薦)
package com.rumenz.task.single; public class SingletonExample3 { //私有構造函數 private SingletonExample3(){ //初始化操作 } private static SingletonExample3 singletonExample3=null; //靜態的工廠方法 public static synchronized SingletonExample3 getSingletonExample3(){ if(singletonExample3==null){ singletonExample3=new SingletonExample3(); } return singletonExample3; } }
缺點:每次進入getSingletonExample3都會加鎖,耗費資源,故不推薦使用。
枚舉(推薦)
package com.rumenz.task.single; public class SingletonExample4 { //私有構造函數 private SingletonExample4(){ //初始化 } public static SingletonExample4 getSingletonExample4(){ return Singleton.INSTANCE.getSingleton(); } private enum Singleton{ INSTANCE; private SingletonExample4 singleton; Singleton(){ singleton=new SingletonExample4(); } public SingletonExample4 getSingleton(){ return singleton; } } }
優點:天然線程安全,可防止反射生成實例,推薦使用
以上就是Java 如何安全的發佈對象的詳細內容,更多關於Java 安全的發佈對象的資料請關註WalkonNet其它相關文章!