成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

單例模式的這些細節你都知道么?

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

[[358261]]

 本文轉載自微信公眾號「丙丙」,作者丙丙 。轉載本文請聯系丙丙公眾號。

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

也經常聽到身邊的同學說身邊誰面試題目很簡單,自己也會,但是真到自己的時候你能就一個知識點講的很透徹,并且能夠發散思考引出更多的答案嗎?

這道手寫一個單例模式的筆試題,應該很多程序員老鐵都碰到過,剛看到題目的時候可能很驚喜,覺得自己運氣真好,但是你真的能夠100%做好這道題嗎?

請手寫一個單例,要求線程安全。

首先給單例下一個定義:在當前進程中,通過單例模式創建的類有且只有一個實例。

單例有如下幾個特點:

  • 在Java應用中,單例模式能保證在一個JVM中,該對象只有一個實例存在
  • 構造器必須是私有的,外部類無法通過調用構造器方法創建該實例
  • 沒有公開的set方法,外部類無法調用set方法創建該實例
  • 提供一個公開的get方法獲取唯一的這個實例

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

某些類創建比較頻繁,對于一些大型的對象,這是一筆很大的系統開銷

省去了new操作符,降低了系統內存的使用頻率,減輕GC壓力

系統中某些類,如spring里的controller,控制著處理流程,如果該類可以創建多個的話,系統完全亂了

好了,單例模式的定義也清楚了,好處也了解了,先看一個餓漢式的寫法

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

如果面試時提供的是這個答案,那丙丙建議你先回家埋頭苦讀兩個月再出來找工作了,你肯定說這個也太low了,一看就是線程不安全的,方法應該通過synchronized給鎖起來,同時創建前先校驗一下,改造后寫法如下:

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

這是一種典型的時間換空間的寫法,不管三七二十一,每次創建實例時先鎖起來,再進行判斷,嚴重降低了系統的處理速度。

有沒有更好的處理方式呢?

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

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

將synchronized關鍵字加在了內部,也就是說當調用的時候是不需要加鎖的,只有在instance為null,并創建對象的時候才需要加鎖,性能有一定的提升。

但是,這樣就沒有問題了嗎?

看下面的情況:在Java指令中創建對象和賦值操作是分開進行的,也就是說instance = new Singleton();語句是分兩步執行的。

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

這樣就可能出錯了,我們以A、B兩個線程為例:

  • A、B線程同時進入了第一個if判斷
  • A首先進入synchronized塊,由于instance為null,所以它執行instance = new Singleton();
  • 由于JVM內部的優化機制,JVM先畫出了一些分配給Singleton實例的空白內存,并賦值給instance成員(注意此時JVM沒有開始初始化這個實例),然后A離開了synchronized塊。

image-20201212010622553

  • B進入synchronized塊,由于instance此時不是null,因此它馬上離開了synchronized塊并將結果返回給調用該方法的程序。
  • 此時B線程打算使用Singleton實例,卻發現它沒有被初始化,于是錯誤發生了。

加上volatile修飾Singleton,再做一次優化:

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

**通過volatile修飾的變量,不會被線程本地緩存,所有線程對該對象的讀寫都會第一時間同步到主內存,從而保證多個線程間該對象的準確性 **

volatile的作用

防止指令重排序,因為instance = new Singleton()不是原子操作

保證內存可見

這個是比較完美的寫法了,這種方式能夠安全的創建唯一的一個實例,又不會對性能有太大的影響。

但是由于volatile關鍵字可能會屏蔽掉虛擬機中一些必要的代碼優化,所以運行效率并不是很高,還有更優的寫法嗎?

通過靜態內部類

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

使用內部類來維護單例的實現,JVM內部的機制能夠保證當一個類被加載的時候,這個類的加載過程是線程互斥的。

這樣當我們第一次調用getInstance的時候,JVM能夠幫我們保證instance只被創建一次,并且會保證把賦值給instance的內存初始化完畢, 這樣我們就不用擔心上面的問題。

同時該方法也只會在第一次調用的時候使用互斥機制,這樣就解決了低性能問題。這樣我們暫時總結一個完美的單例模式。

還有更完美的寫法嗎,通過枚舉:

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

使用枚舉來實現單實例控制會更加簡潔,而且JVM從根本上提供保障,絕對防止多次實例化,是更簡潔、高效、安全的實現單例的方式。

 

責任編輯:武曉燕 來源: 丙丙
相關推薦

2023-02-26 23:33:02

SQLMySQL數據庫

2023-12-05 08:20:05

單例模式Python

2019-02-12 11:15:15

Spring設計模式Java

2023-10-09 08:31:19

2021-06-04 10:11:07

鴻蒙安卓操作系統

2018-04-03 15:38:07

Java單例模式模式設計

2020-04-27 08:31:29

單例模式Python軟件設計模式

2021-08-05 18:21:29

Autowired代碼spring

2023-01-16 08:09:51

SpringMVC句柄

2016-03-18 19:03:35

認知計算IBM

2022-11-10 09:00:41

2009-10-15 13:48:13

服務器維護常識

2019-01-03 14:30:04

數據庫優化索引

2019-07-08 10:18:38

MPLSIP數據

2016-01-11 09:48:07

2024-03-26 10:10:45

JavaScript操作符操作表達式

2022-09-07 09:01:14

JS操作符運算符

2020-05-27 11:30:54

Chrome DevT前端命令

2021-07-29 06:55:03

Spring@AutowriedbyType注入

2018-05-20 11:01:47

Siri語音助手手機
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 久久视频精品在线 | 国产精品久久久久久久久污网站 | 国产成人精品一区二 | 亚洲国产精品一区二区www | 亚洲成人一区二区 | 国产激情一区二区三区 | 亚洲精品一区二区在线 | 亚洲一区二区三区免费在线观看 | 久久国产精品久久国产精品 | 欧美精品一区在线 | 亚洲精品久久久久中文字幕欢迎你 | 人人干在线| 中文字幕在线视频网站 | 亚洲免费网站 | 亚洲日韩中文字幕一区 | 五月婷婷中文 | 欧美一区二区三区,视频 | 宅女噜噜66国产精品观看免费 | 国产高清精品一区二区三区 | 亚洲第一网站 | 亚洲精品欧美 | 在线视频一区二区 | 亚洲欧洲中文日韩 | 国产三区视频在线观看 | 午夜视频在线免费观看 | 国产网站在线免费观看 | 成人性生交大片免费看中文带字幕 | 国产精品久久亚洲 | 高清色 | 黑人精品 | 欧美日韩久久精品 | 羞羞视频在线观免费观看 | 国产日韩一区二区三免费高清 | 日本天天操 | 亚洲精视频 | 久久久一二三 | 国产精品久久一区二区三区 | 黑人巨大精品欧美一区二区免费 | 日本国产一区二区 | 玖玖视频| 91精品久久久久 |