<strike id="ca4is"><em id="ca4is"></em></strike>
  • <sup id="ca4is"></sup>
    • <s id="ca4is"><em id="ca4is"></em></s>
      <option id="ca4is"><cite id="ca4is"></cite></option>
    • 二維碼
      企資網

      掃一掃關注

      當前位置: 首頁 » 企業資訊 » 熱點 » 正文

      單例模式,關鍵字級別詳解

      放大字體  縮小字體 發布日期:2021-09-01 05:50:36    作者:企資小編    瀏覽次數:118
      導讀

      大家好,我是課代表。歡迎關注我的公眾號_Java課代表,原創實戰干貨首發地兒,不要錯過呦!來,我們開始今天的分享!0.前言如果你去問一個寫過幾年代碼的程序員用過哪些設計模式,我打賭,90_以上的回答里面會帶【單

      大家好,我是課代表。歡迎關注我的公眾號_Java課代表,原創實戰干貨首發地兒,不要錯過呦!

      來,我們開始今天的分享!

      0.前言

      如果你去問一個寫過幾年代碼的程序員用過哪些設計模式,我打賭,90_以上的回答里面會帶【單例模式】。甚至有的面試官會直接問_說一下你用過哪些設計模式,單例就不用說了。你看,連面試官都聽煩了,火爆程度可見一斑。

      不過,看似簡單的單例模式,里面蘊含了很多Java基礎,日常開發過程中課代表見過很多不規范的,甚至是有問題的單例實現。所以整理此文,總結一下單例模式的最佳實踐。

      1、懶加載(懶漢)

      所謂懶加載,就是直到第一次被調用時才加載。其實現需要考慮并發問題和指令重排,代碼如下_

      public class Singleton {    private volatile static Singleton instance; //①    private Singleton() { //②    }    public static Singleton getInstance() {        if (instance __ null) {//③            synchronized (Singleton.class) {                if (instance __ null) {//④                    instance _ new Singleton();//⑤                }            }        }        return instance;    }}

      這段代碼精簡至極,沒有一個字符是多余的,下面逐行解讀一下_

      首先,注意到①處的volatile關鍵字,她具備兩項特性_

      一是保證此變量對于所有線程的可見性。即當一條線程修改了這個變量的值,新值對于其他線程來說是可以立即得知的。

      二是禁止指令重排序優化。

      這里解釋一下指令重排序優化_

      代碼 ⑤ 處的instance _ new Singleton();并不是原子的,大體可分為如下 3 步_

        分配內存調用構造函數初始化成實例
      1. instance指向分配的內存空間

      JVM 允許在保證結果正確的前提下進行指令重排序優化。即如上 3 步可能的順序為1->2->3 或 1->3->2 。如果順序是 1->3->2 ,當 3 執行完,2 還未執行時,另一個線程執行到代碼 ③ 處,發現instance不為null,直接返回還未初始化好的instance并使用,就會報錯。

      所以使用volatile,就是為了保證線程間的可見性和防止指令重排。

      其次,代碼②處將構造函數聲明為private目的在于阻止使用new Singleton()這樣的代碼生成新實例。

      最后,當客戶端調用Singleton.getInstance()時,先檢查是否已經實例化(代碼③),未實例化時同步代碼塊,然后再次檢查是否已實例化(代碼④),然后才執行代碼⑤。兩次檢查的意義在于,防止synchronized同步過程中其他線程進行了實例化。

      這就是著名的雙重檢查鎖(Double check lock)實現單例,也即懶加載。

      TIPS:

      網上也有直接對getInstance()方法加鎖的版本,這樣大范圍的方法級別加鎖會導致并發變低,實際上第一次調用生成實例之后,后續獲取實例根本不需要并發控制了。而本例的雙重檢查鎖版本可以避免此并發問題。

      2、預加載(餓漢)

      與懶加載相對應,預加載是在類加載時就已經初始化好了,所以是天然線程安全的,代碼如下_

      public class Singleton {    private static final Singleton instance _ new Singleton();// ①        private Singleton(){}        public static Singleton getInstance(){        return instance;    }}

      注意到 ① 處的類變量使用了final

      這里用final更多的意義在于提供語法約束。畢竟你是單例,就只有這一個實例,不可能再指向另一個。instance有了final的約束,后面再有人不小心編寫了修改其指向的代碼就會報語法錯誤。

      這就好比@Override注解,你能保證寫對方法名和參數,那不寫注解也沒問題,但是有了注解的約束,編譯器就會幫你檢查,還能防止別人亂改。

      3、靜態內部類

      此方法和預加載原理相同,都是利用JVM類加載的特性實現天然的線程安全,不同之處在于,靜態內部類做到了延遲加載。

      public class Singleton {        private static class SingletonHolder {        private static Singleton instance _ new Singleton();    }        private Singleton(){}    public static Singleton getInstance() {        return SingletonHolder.instance;    }}

      SingletonHolder 是靜態內部類,當外部類Singleton被加載的時候并不會創建任何實例,只有當Singleton.getInstance()被調用的時候,才會創建Singleton實例,這一切由 JVM 天然完成,所以既保證了線程安全,又實現了延遲加載。

      4、枚舉

      沒錯,枚舉可以實現單例,而且這種方式是《Effective Java中文版》第二版 中的推薦實現方式。代碼極其簡單_

      public enum Singleton {        INSTANCE;    public void doSomeThing(){        System.out.println("done");    }}

      使用時直接Singleton.INSTANCE.doSomeThing();即可。

      這里主要利用了枚舉的如下兩個特性_

      枚舉的構造器總是私有的,所以不必像前幾種方式一樣顯式定義私有構造方法
    • 枚舉類中的每個值,都是實例(只有INSTANCE這一個實例)

      除此之外,枚舉還附帶了一些額外好處_無償地提供了序列化機制,還可以防止通過多次反序列化生成多個實例。

      鑒于此,單例的最佳實踐就是用枚舉來實現。

      5、總結

      事實上,單例的寫法并不止于本文所提的這 4 種,你可能還會看到很多其他變種,她們或多或少都存在一些缺陷,比如,懶加載方式將synchronized作用于整個方法上也能實現,但頻繁加鎖,釋放鎖會產生性能瓶頸,而完全去掉鎖又會帶來并發問題。

      所以,只要吃透了文中列出的這 4 種單例方式,就能做到舉一反三,見到別人寫的單例也能一眼看出對錯。

      文中所列的 4 種單例模式,除了枚舉之外,全都用到了static關鍵字,《Java 虛擬機規范》 規定,有幾種情況必須立即對類進行“初始化”,其中涉及static的場景如下_

      讀取或設置一個類型的靜態字段(被 final 修飾、已在編譯期把結果放入常量池的靜態字段除外)的時候。

      調用一個類型的靜態方法的時候。

      懶加載,預加載和靜態內部類正是利用了這兩點特性。

      static關鍵字遺忘的同學可以參看我的另一篇文章_《一題搞定static關鍵字》

      最后,再次強調一下,如果大家開發中需要手寫單例,建議聽從 Joshua Bloch在《Effective Java中文版》第二版 中的建議_

      單元素的枚舉類型已經成為實現 Singleton 的最佳方法

      參考資料_

      1、《Effective Java中文版》 Joshua Bloch 第二版 P15

      2、《深入理解 Java 虛擬機》 周志明 第3版,P444-P448,P264

      3、深入淺出單實例SINGLETON設計模式

      相關原創推薦

      一題搞定static關鍵字

    •  
      (文/企資小編)
      免責聲明
      本文僅代表作發布者:企資小編個人觀點,本站未對其內容進行核實,請讀者僅做參考,如若文中涉及有違公德、觸犯法律的內容,一經發現,立即刪除,需自行承擔相應責任。涉及到版權或其他問題,請及時聯系我們刪除處理郵件:weilaitui@qq.com。
       

      Copyright ? 2016 - 2025 - 企資網 48903.COM All Rights Reserved 粵公網安備 44030702000589號

      粵ICP備16078936號

      微信

      關注
      微信

      微信二維碼

      WAP二維碼

      客服

      聯系
      客服

      聯系客服:

      在線QQ: 303377504

      客服電話: 020-82301567

      E_mail郵箱: weilaitui@qq.com

      微信公眾號: weishitui

      客服001 客服002 客服003

      工作時間:

      周一至周五: 09:00 - 18:00

      反饋

      用戶
      反饋

      午夜久久久久久网站,99久久www免费,欧美日本日韩aⅴ在线视频,东京干手机福利视频
        <strike id="ca4is"><em id="ca4is"></em></strike>
      • <sup id="ca4is"></sup>
        • <s id="ca4is"><em id="ca4is"></em></s>
          <option id="ca4is"><cite id="ca4is"></cite></option>
        • 主站蜘蛛池模板: 国产精品毛片一区二区| 精品无码国产一区二区三区av| 狠狠色香婷婷久久亚洲精品| 最新国产午夜精品视频不卡| 天天操夜夜操免费视频| 国产人妖ts视频在线观看| 亚洲欧美精品一区天堂久久| 中文字幕第15页| free哆啪啪免费永久| 王爷晚上含奶h嗯额嗯| 日产国语一区二区三区在线看| 国产精品电影一区二区| 制服丝袜在线不卡| 久久人人妻人人做人人爽| 思99热精品久久只有精品| 狠狠色噜噜狠狠狠888米奇视频| 无码免费一区二区三区免费播放| 国产精品免费精品自在线观看 | 一级做a爰片久久毛片唾| 4虎永免费最新永久免费地址| 波多野结衣伦理片在线观看| 成av免费大片黄在线观看| 国产在线国偷精品产拍| 亚洲人免费视频| 99riav国产在线观看| 精品人妻系列无码一区二区三区| 日本人成动漫网站在线观看| 国产精品19禁在线观看2021| 亚洲精品无码不卡| 《波多野结衣系列mkmp-305》| 蜜桃视频无码区在线观看| 最新中文字幕在线播放| 国产精品小青蛙在线观看| 亚洲综合第一区| xxxxx性bbbbb欧美| 精品福利三区3d卡通动漫| 无码精品一区二区三区免费视频| 国产在线高清精品二区色五郎| 亚洲av无码之日韩精品| 两个人看的视频高清在线www| 欧美高清一区二区三区|