Java 如何安全的發(fā)布對象
安全發(fā)布對象
- 在靜態(tài)初始化函數(shù)中初始化一個對象引用
- 將對象的引用保存到volatile類型域或者AtomicReference對象中
- 將對象的引用保存到某個正確構(gòu)造對象的final類型域中
- 將對象的引用保存到一個由鎖保護的域中
Spring 框架中,Spring管理的類都是單例模式。如何保證一個實例只被初始化一次,且線程安全?通過不同單例的寫法,具體描述安全發(fā)布對象的四種方法:
在靜態(tài)初始化函數(shù)中初始化一個對象的引用(不推薦)
package com.rumenz.task.single; //線程安全 //餓漢模式 //靜態(tài)代碼塊初始化 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; //線程安全 //餓漢模式 //靜態(tài)代碼塊初始化 public class SingletonExample { private SingletonExample(){ //初始化操作 } private static SingletonExample singletonExample=new SingletonExample(); public static SingletonExample getInstance(){ return singletonExample; } }
缺點:用不用都會初始化對象,如果初始化工作較多,加載速度會變慢,影響系統(tǒng)性能。
將對象的引用保存到volatile類型或AtomicReference對象中(推薦)
package com.rumenz.task.single; //線程安全 //懶漢模式 public class SingletonExample1 { private SingletonExample1() { //初始化操作 } // 1、memory = allocate() 分配對象的內(nèi)存空間 // 2、ctorInstance() 初始化對象 // 3、instance = memory 設(shè)置instance指向剛分配的內(nèi)存 // 單例對象 volatile + 雙重檢測機制 -> 禁止指令重排 private volatile static SingletonExample1 singletonExample1=null; //靜態(tài)工廠方法 public static SingletonExample1 getInstance(){ if(singletonExample1==null){ //雙重檢測 synchronized(SingletonExample1.class){ //同步鎖 if(singletonExample1==null){ singletonExample1=new SingletonExample1(); } } } return singletonExample1; } }
優(yōu)點:按需加載
缺點:第一次初始化的時候可能會比較慢
通過synchronized(不推薦)
package com.rumenz.task.single; public class SingletonExample3 { //私有構(gòu)造函數(shù) private SingletonExample3(){ //初始化操作 } private static SingletonExample3 singletonExample3=null; //靜態(tài)的工廠方法 public static synchronized SingletonExample3 getSingletonExample3(){ if(singletonExample3==null){ singletonExample3=new SingletonExample3(); } return singletonExample3; } }
缺點:每次進入getSingletonExample3都會加鎖,耗費資源,故不推薦使用。
枚舉(推薦)
package com.rumenz.task.single; public class SingletonExample4 { //私有構(gòu)造函數(shù) 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; } } }
優(yōu)點:天然線程安全,可防止反射生成實例,推薦使用
以上就是Java 如何安全的發(fā)布對象的詳細內(nèi)容,更多關(guān)于Java 安全的發(fā)布對象的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
SpringBoot+layuimini實現(xiàn)左側(cè)菜單動態(tài)展示的示例代碼
Layuimini是Layui的升級版,它是專業(yè)做后臺頁面的框架,而且是適合PC端和移動端,以下地址可以在PC端顯示,也可以在手機上顯示,只不過會做自適應(yīng),本文將給大家介紹了SpringBoot+layuimini實現(xiàn)左側(cè)菜單動態(tài)展示的方法,需要的朋友可以參考下2024-04-04springboot如何解決跨域后session獲取不到sessionId不一致
這篇文章主要介紹了springboot如何解決跨域后session獲取不到sessionId不一致問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-01-01