Java多線程實(shí)戰(zhàn)之單例模式與多線程的實(shí)例詳解
1、立即加載/餓漢模式
// 立即加載/餓漢模式
public class MyObject {
private static final MyObject myObject = new MyObject();
private MyObject() {
}
public static MyObject getInstance() {
return myObject;
}
}
立即加載/餓漢模式是在類創(chuàng)建的同時(shí)已經(jīng)創(chuàng)建好一個(gè)靜態(tài)的對(duì)象供系統(tǒng)使用,不存在線程安全問(wèn)題
2、延遲加載/懶漢模式
// 延遲加載/懶漢模式
public class MyObject {
private static MyObject myObject;
private MyObject() {
}
public static MyObject getInstance() {
if (myObject == null) {
myObject = new MyObject();
}
return myObject;
}
}
延遲加載/懶漢模式是在調(diào)用方法時(shí)實(shí)例才被創(chuàng)建,在多線程環(huán)境下,會(huì)出現(xiàn)取出多個(gè)實(shí)例的情況,與單例模式的初衷是相背離的
1)、延遲加載/懶漢模式在多線程環(huán)境下創(chuàng)建出多個(gè)實(shí)例:
// 延遲加載/懶漢模式
public class MyObject {
private static MyObject myObject;
private MyObject() {
}
public static MyObject getInstance() {
try {
if (myObject == null) {
TimeUnit.SECONDS.sleep(3);
myObject = new MyObject();
}
} catch (InterruptedException e) {
e.printStackTrace();
}
return myObject;
}
}
public class MyThread extends Thread {
@Override
public void run() {
System.out.println(MyObject.getInstance().hashCode());
}
}
public class Run {
public static void main(String[] args) {
MyThread myThread = new MyThread();
MyThread myThread2 = new MyThread();
MyThread myThread3 = new MyThread();
myThread.start();
myThread2.start();
myThread3.start();
}
}
運(yùn)行結(jié)果:三次打印的hashCode不完全相等
2)、通過(guò)聲明synchronized關(guān)鍵字解決線程安全問(wèn)題:
// 延遲加載/懶漢模式
public class MyObject {
private static MyObject myObject;
private MyObject() {
}
public static synchronized MyObject getInstance() {
try {
if (myObject == null) {
TimeUnit.SECONDS.sleep(3);
myObject = new MyObject();
}
} catch (InterruptedException e) {
e.printStackTrace();
}
return myObject;
}
}
使用synchronized關(guān)鍵字,這種方法的運(yùn)行效率很低,是同步運(yùn)行的,下一個(gè)線程想要取得對(duì)象,則必須等上一個(gè)線程釋放鎖之后,才可以繼續(xù)執(zhí)行
3)、使用同步代碼塊解決線程安全問(wèn)題:
// 延遲加載/懶漢模式
public class MyObject {
private static MyObject myObject;
private MyObject() {
}
public static MyObject getInstance() {
try {
// 相當(dāng)于public static synchronized MyObject getInstance()
synchronized (MyObject.class) {
if (myObject == null) {
TimeUnit.SECONDS.sleep(3);
myObject = new MyObject();
}
}
} catch (InterruptedException e) {
e.printStackTrace();
}
return myObject;
}
}
加入同步代碼塊,這種方法的運(yùn)行效率也是非常低,和synchronized同步方法一樣是同步運(yùn)行的
4)、針對(duì)某些重要的代碼進(jìn)行單獨(dú)的同步
// 延遲加載/懶漢模式
public class MyObject {
private static MyObject myObject;
private MyObject() {
}
public static MyObject getInstance() {
try {
if (myObject == null) {
TimeUnit.SECONDS.sleep(3);
synchronized (MyObject.class) {
myObject = new MyObject();
}
}
} catch (InterruptedException e) {
e.printStackTrace();
}
return myObject;
}
}
此方法只對(duì)實(shí)例化對(duì)象的關(guān)鍵代碼進(jìn)行同步,從語(yǔ)句的結(jié)構(gòu)上來(lái)講,運(yùn)行的效率的確得到了提升。但如果是多線程的情況下還是無(wú)法解決得到同一個(gè)實(shí)例對(duì)象的結(jié)果
5)、使用DCL雙檢查鎖機(jī)制
// 延遲加載/懶漢模式
public class MyObject {
private volatile static MyObject myObject;
private MyObject() {
}
public static MyObject getInstance() {
try {
if (myObject == null) {
TimeUnit.SECONDS.sleep(3);
synchronized (MyObject.class) {
if (myObject == null) {
myObject = new MyObject();
}
}
}
} catch (InterruptedException e) {
e.printStackTrace();
}
return myObject;
}
}
使用DCL雙檢查鎖機(jī)制,既保證了不需要同步代碼的異步執(zhí)行性,又保證了單例的效果
3、使用靜態(tài)內(nèi)部類實(shí)現(xiàn)單例模式
public class MyObject {
private static class MyObjectHandler {
private static MyObject myObject = new MyObject();
}
private MyObject() {
}
public static MyObject getInstance() {
return MyObjectHandler.myObject;
}
}
4、使用靜態(tài)代碼塊實(shí)現(xiàn)單例模式
public class MyObject {
private static MyObject instance = null;
private MyObject() {
}
static {
instance = new MyObject();
}
public static MyObject getInstance() {
return instance;
}
}
5、使用enum枚舉實(shí)現(xiàn)單例模式
public class MyObject {
public enum MyEnumSingleton {
objectFactory;
private MyObject myObject;
private MyEnumSingleton() {
myObject = new MyObject();
}
public MyObject getInstance() {
return myObject;
}
}
public static MyObject getInstance() {
return MyEnumSingleton.objectFactory.getInstance();
}
}
枚舉enum和靜態(tài)代碼塊的特性相似,在使用枚舉類時(shí),構(gòu)造方法會(huì)被自動(dòng)調(diào)用,使用這個(gè)特性實(shí)現(xiàn)單例設(shè)計(jì)模式
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,謝謝大家對(duì)腳本之家的支持。如果你想了解更多相關(guān)內(nèi)容請(qǐng)查看下面相關(guān)鏈接
相關(guān)文章
IDEA?報(bào)Plugin'maven-resources-plugin:'not?found?
如果在使用?IDEA?時(shí)遇到?"Plugin?'maven-resources-plugin:'?not?found"?錯(cuò)誤,可能是由于?Maven?倉(cāng)庫(kù)中未找到所需的?Maven?插件,近小編給大家分享幾種解決方法,感興趣的朋友跟隨小編一起看看吧2023-07-07
intellij idea創(chuàng)建第一個(gè)動(dòng)態(tài)web項(xiàng)目的步驟方法
這篇文章主要介紹了intellij idea創(chuàng)建第一個(gè)動(dòng)態(tài)web項(xiàng)目的步驟方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-10-10
MAC算法之消息摘要算法HmacMD5的實(shí)現(xiàn)
這篇文章主要介紹了MAC算法之消息摘要算法HmacMD5的實(shí)現(xiàn)的相關(guān)資料,這里提供實(shí)例,幫助大家學(xué)習(xí)理解這部分知識(shí),需要的朋友可以參考下2017-08-08
Mybatis-plus動(dòng)態(tài)條件查詢QueryWrapper的使用案例
mybatis-plus框架功能很強(qiáng)大,把很多功能都集成了,下面這篇文章主要給大家介紹了關(guān)于Mybatis-plus動(dòng)態(tài)條件查詢QueryWrapper的使用教程,文中通過(guò)圖文介紹的非常詳細(xì),需要的朋友可以參考下2022-07-07
Java導(dǎo)出excel時(shí)合并同一列中相同內(nèi)容的行思路詳解
這篇文章主要介紹了Java導(dǎo)出excel時(shí)合并同一列中相同內(nèi)容的行,需要的朋友可以參考下2018-06-06
多模塊項(xiàng)目使用枚舉配置spring-cache緩存方案詳解
這篇文章主要為大家介紹了多模塊項(xiàng)目使用枚舉配置spring-cache緩存的方案詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-05-05
Java?ArrayList實(shí)現(xiàn)刪除指定位置的元素
目標(biāo):list中有0到39共40個(gè)元素,刪除其中索引是10、20、30的元素。本文為大家整理了三個(gè)不同的方法,感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2023-01-01
解讀SpringBoot中addCorsMappings配置跨域與攔截器互斥問(wèn)題的原因
這篇文章主要介紹了解讀SpringBoot中addCorsMappings配置跨域與攔截器互斥問(wèn)題的原因,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-12-12

