自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

單例模式的這些細(xì)節(jié)你都知道么?

開(kāi)發(fā) 后端
從去年開(kāi)始,Java程序員們換工作面試越來(lái)越難,不懂多線(xiàn)程,JVM,mysql,以及一些分布式的組件都不好意思出去面試。

[[358261]]

 本文轉(zhuǎn)載自微信公眾號(hào)「丙丙」,作者丙丙 。轉(zhuǎn)載本文請(qǐng)聯(lián)系丙丙公眾號(hào)。

從去年開(kāi)始,Java程序員們換工作面試越來(lái)越難,不懂多線(xiàn)程,JVM,mysql,以及一些分布式的組件都不好意思出去面試。

也經(jīng)常聽(tīng)到身邊的同學(xué)說(shuō)身邊誰(shuí)面試題目很簡(jiǎn)單,自己也會(huì),但是真到自己的時(shí)候你能就一個(gè)知識(shí)點(diǎn)講的很透徹,并且能夠發(fā)散思考引出更多的答案嗎?

這道手寫(xiě)一個(gè)單例模式的筆試題,應(yīng)該很多程序員老鐵都碰到過(guò),剛看到題目的時(shí)候可能很驚喜,覺(jué)得自己運(yùn)氣真好,但是你真的能夠100%做好這道題嗎?

請(qǐng)手寫(xiě)一個(gè)單例,要求線(xiàn)程安全。

首先給單例下一個(gè)定義:在當(dāng)前進(jìn)程中,通過(guò)單例模式創(chuàng)建的類(lèi)有且只有一個(gè)實(shí)例。

單例有如下幾個(gè)特點(diǎn):

  • 在Java應(yīng)用中,單例模式能保證在一個(gè)JVM中,該對(duì)象只有一個(gè)實(shí)例存在
  • 構(gòu)造器必須是私有的,外部類(lèi)無(wú)法通過(guò)調(diào)用構(gòu)造器方法創(chuàng)建該實(shí)例
  • 沒(méi)有公開(kāi)的set方法,外部類(lèi)無(wú)法調(diào)用set方法創(chuàng)建該實(shí)例
  • 提供一個(gè)公開(kāi)的get方法獲取唯一的這個(gè)實(shí)例

那單例模式有什么好處呢?

某些類(lèi)創(chuàng)建比較頻繁,對(duì)于一些大型的對(duì)象,這是一筆很大的系統(tǒng)開(kāi)銷(xiāo)

省去了new操作符,降低了系統(tǒng)內(nèi)存的使用頻率,減輕GC壓力

系統(tǒng)中某些類(lèi),如spring里的controller,控制著處理流程,如果該類(lèi)可以創(chuàng)建多個(gè)的話(huà),系統(tǒng)完全亂了

好了,單例模式的定義也清楚了,好處也了解了,先看一個(gè)餓漢式的寫(xiě)法

  1. public class Singleton { 
  2.     private static Singleton instance = new Singleton(); 
  3.     /** 
  4.      * 私有構(gòu)造方法,防止被實(shí)例化 
  5.      */ 
  6.     private Singleton(){} 
  7.     /** 
  8.      * 靜態(tài)get方法 
  9.      */ 
  10.     public static Singleton getInstance(){ 
  11.         return instance; 
  12.     } 

如果面試時(shí)提供的是這個(gè)答案,那丙丙建議你先回家埋頭苦讀兩個(gè)月再出來(lái)找工作了,你肯定說(shuō)這個(gè)也太low了,一看就是線(xiàn)程不安全的,方法應(yīng)該通過(guò)synchronized給鎖起來(lái),同時(shí)創(chuàng)建前先校驗(yàn)一下,改造后寫(xiě)法如下:

  1. public class Singleton { 
  2.     private static Singleton instance = null
  3.     /** 
  4.      * 私有構(gòu)造方法,防止被實(shí)例化 
  5.      */ 
  6.     private Singleton(){} 
  7.     /** 
  8.      * 靜態(tài)get方法 
  9.      */ 
  10.     public static synchronized Singleton getInstance(){ 
  11.         if(instance == null){ 
  12.             instance = new Singleton(); 
  13.         } 
  14.         return instance; 
  15.     } 

這是一種典型的時(shí)間換空間的寫(xiě)法,不管三七二十一,每次創(chuàng)建實(shí)例時(shí)先鎖起來(lái),再進(jìn)行判斷,嚴(yán)重降低了系統(tǒng)的處理速度。

有沒(méi)有更好的處理方式呢?

有,通過(guò)雙檢鎖做兩次判斷,代碼如下:

  1. public class Singleton { 
  2.     private static Singleton instance = null
  3.     private Singleton(){} 
  4.     public static Singleton getInstance(){ 
  5.         //先檢查實(shí)例是否存在,如果不存在才進(jìn)入下面的同步塊 
  6.         if(instance == null){ 
  7.             //同步塊,線(xiàn)程安全的創(chuàng)建實(shí)例 
  8.             synchronized (Singleton.class) { 
  9.                 //再次檢查實(shí)例是否存在,如果不存在才真正的創(chuàng)建實(shí)例 
  10.                 if(instance == null){ 
  11.                     instance = new Singleton(); 
  12.                 } 
  13.             } 
  14.         } 
  15.         return instance; 
  16.     } 

將synchronized關(guān)鍵字加在了內(nèi)部,也就是說(shuō)當(dāng)調(diào)用的時(shí)候是不需要加鎖的,只有在instance為null,并創(chuàng)建對(duì)象的時(shí)候才需要加鎖,性能有一定的提升。

但是,這樣就沒(méi)有問(wèn)題了嗎?

看下面的情況:在Java指令中創(chuàng)建對(duì)象和賦值操作是分開(kāi)進(jìn)行的,也就是說(shuō)instance = new Singleton();語(yǔ)句是分兩步執(zhí)行的。

但是JVM并不保證這兩個(gè)操作的先后順序,也就是說(shuō)有可能JVM會(huì)為新的Singleton實(shí)例分配空間,然后直接賦值給instance成員,然后再去初始化這個(gè)Singleton實(shí)例。

這樣就可能出錯(cuò)了,我們以A、B兩個(gè)線(xiàn)程為例:

  • A、B線(xiàn)程同時(shí)進(jìn)入了第一個(gè)if判斷
  • A首先進(jìn)入synchronized塊,由于instance為null,所以它執(zhí)行instance = new Singleton();
  • 由于JVM內(nèi)部的優(yōu)化機(jī)制,JVM先畫(huà)出了一些分配給Singleton實(shí)例的空白內(nèi)存,并賦值給instance成員(注意此時(shí)JVM沒(méi)有開(kāi)始初始化這個(gè)實(shí)例),然后A離開(kāi)了synchronized塊。

image-20201212010622553

  • B進(jìn)入synchronized塊,由于instance此時(shí)不是null,因此它馬上離開(kāi)了synchronized塊并將結(jié)果返回給調(diào)用該方法的程序。
  • 此時(shí)B線(xiàn)程打算使用Singleton實(shí)例,卻發(fā)現(xiàn)它沒(méi)有被初始化,于是錯(cuò)誤發(fā)生了。

加上volatile修飾Singleton,再做一次優(yōu)化:

  1. public class Singleton { 
  2.     private volatile static Singleton instance = null
  3.     private Singleton(){} 
  4.     public static Singleton getInstance(){ 
  5.         //先檢查實(shí)例是否存在,如果不存在才進(jìn)入下面的同步塊 
  6.         if(instance == null){ 
  7.             //同步塊,線(xiàn)程安全的創(chuàng)建實(shí)例 
  8.             synchronized (Singleton.class) { 
  9.                 //再次檢查實(shí)例是否存在,如果不存在才真正的創(chuàng)建實(shí)例 
  10.                 if(instance == null){ 
  11.                     instance = new Singleton(); 
  12.                 } 
  13.             } 
  14.         } 
  15.         return instance; 
  16.     } 

**通過(guò)volatile修飾的變量,不會(huì)被線(xiàn)程本地緩存,所有線(xiàn)程對(duì)該對(duì)象的讀寫(xiě)都會(huì)第一時(shí)間同步到主內(nèi)存,從而保證多個(gè)線(xiàn)程間該對(duì)象的準(zhǔn)確性 **

volatile的作用

防止指令重排序,因?yàn)閕nstance = new Singleton()不是原子操作

保證內(nèi)存可見(jiàn)

這個(gè)是比較完美的寫(xiě)法了,這種方式能夠安全的創(chuàng)建唯一的一個(gè)實(shí)例,又不會(huì)對(duì)性能有太大的影響。

但是由于volatile關(guān)鍵字可能會(huì)屏蔽掉虛擬機(jī)中一些必要的代碼優(yōu)化,所以運(yùn)行效率并不是很高,還有更優(yōu)的寫(xiě)法嗎?

通過(guò)靜態(tài)內(nèi)部類(lèi)

  1. public class Singleton {   
  2.    
  3.     /* 私有構(gòu)造方法,防止被實(shí)例化 */   
  4.     private Singleton() {   
  5.     }   
  6.    
  7.     /* 此處使用一個(gè)內(nèi)部類(lèi)來(lái)維護(hù)單例 */   
  8.     private static class SingletonFactory {   
  9.         private static Singleton instance = new Singleton();   
  10.     }   
  11.    
  12.     /* 獲取實(shí)例 */   
  13.     public static Singleton getInstance() {   
  14.         return SingletonFactory.instance;   
  15.     }   
  16.    
  17.     /* 如果該對(duì)象被用于序列化,可以保證對(duì)象在序列化前后保持一致 */   
  18.     public Object readResolve() {   
  19.         return getInstance();   
  20.     }   
  21. }   

使用內(nèi)部類(lèi)來(lái)維護(hù)單例的實(shí)現(xiàn),JVM內(nèi)部的機(jī)制能夠保證當(dāng)一個(gè)類(lèi)被加載的時(shí)候,這個(gè)類(lèi)的加載過(guò)程是線(xiàn)程互斥的。

這樣當(dāng)我們第一次調(diào)用getInstance的時(shí)候,JVM能夠幫我們保證instance只被創(chuàng)建一次,并且會(huì)保證把賦值給instance的內(nèi)存初始化完畢, 這樣我們就不用擔(dān)心上面的問(wèn)題。

同時(shí)該方法也只會(huì)在第一次調(diào)用的時(shí)候使用互斥機(jī)制,這樣就解決了低性能問(wèn)題。這樣我們暫時(shí)總結(jié)一個(gè)完美的單例模式。

還有更完美的寫(xiě)法嗎,通過(guò)枚舉:

  1. public enum Singleton { 
  2.     /** 
  3.      * 定義一個(gè)枚舉的元素,它就代表了Singleton的一個(gè)實(shí)例。 
  4.      */ 
  5.     Instance; 

使用枚舉來(lái)實(shí)現(xiàn)單實(shí)例控制會(huì)更加簡(jiǎn)潔,而且JVM從根本上提供保障,絕對(duì)防止多次實(shí)例化,是更簡(jiǎn)潔、高效、安全的實(shí)現(xiàn)單例的方式。

 

責(zé)任編輯:武曉燕 來(lái)源: 丙丙
相關(guān)推薦

2023-02-26 23:33:02

SQLMySQL數(shù)據(jù)庫(kù)

2023-12-05 08:20:05

單例模式Python

2019-02-12 11:15:15

Spring設(shè)計(jì)模式Java

2023-10-09 08:31:19

2018-04-03 15:38:07

Java單例模式模式設(shè)計(jì)

2021-06-04 10:11:07

鴻蒙安卓操作系統(tǒng)

2020-04-27 08:31:29

單例模式Python軟件設(shè)計(jì)模式

2021-08-05 18:21:29

Autowired代碼spring

2023-01-16 08:09:51

SpringMVC句柄

2009-10-15 13:48:13

服務(wù)器維護(hù)常識(shí)

2016-03-18 19:03:35

認(rèn)知計(jì)算IBM

2022-11-10 09:00:41

2019-01-03 14:30:04

數(shù)據(jù)庫(kù)優(yōu)化索引

2019-07-08 10:18:38

MPLSIP數(shù)據(jù)

2016-01-11 09:48:07

2024-03-26 10:10:45

JavaScript操作符操作表達(dá)式

2020-05-27 11:30:54

Chrome DevT前端命令

2021-07-29 06:55:03

Spring@AutowriedbyType注入

2022-09-07 09:01:14

JS操作符運(yùn)算符

2022-03-15 09:58:12

單例模式系統(tǒng)
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)