二維碼
        企資網(wǎng)

        掃一掃關(guān)注

        當(dāng)前位置: 首頁(yè) » 企資頭條 » 產(chǎn)業(yè) » 正文

        單例模式,關(guān)鍵字級(jí)別詳解

        放大字體  縮小字體 發(fā)布日期:2021-09-01 05:50:36    作者:企資小編    瀏覽次數(shù):91
        導(dǎo)讀

        大家好,我是課代表。歡迎關(guān)注我的公眾號(hào)_Java課代表,原創(chuàng)實(shí)戰(zhàn)干貨首發(fā)地兒,不要錯(cuò)過(guò)呦!來(lái),我們開(kāi)始今天的分享!0.前言如果你去問(wèn)一個(gè)寫(xiě)過(guò)幾年代碼的程序員用過(guò)哪些設(shè)計(jì)模式,我打賭,90_以上的回答里面會(huì)帶【單

        大家好,我是課代表。歡迎關(guān)注我的公眾號(hào)_Java課代表,原創(chuàng)實(shí)戰(zhàn)干貨首發(fā)地兒,不要錯(cuò)過(guò)呦!

        來(lái),我們開(kāi)始今天的分享!

        0.前言

        如果你去問(wèn)一個(gè)寫(xiě)過(guò)幾年代碼的程序員用過(guò)哪些設(shè)計(jì)模式,我打賭,90_以上的回答里面會(huì)帶【單例模式】。甚至有的面試官會(huì)直接問(wèn)_說(shuō)一下你用過(guò)哪些設(shè)計(jì)模式,單例就不用說(shuō)了。你看,連面試官都聽(tīng)煩了,火爆程度可見(jiàn)一斑。

        不過(guò),看似簡(jiǎn)單的單例模式,里面蘊(yùn)含了很多Java基礎(chǔ),日常開(kāi)發(fā)過(guò)程中課代表見(jiàn)過(guò)很多不規(guī)范的,甚至是有問(wèn)題的單例實(shí)現(xiàn)。所以整理此文,總結(jié)一下單例模式的最佳實(shí)踐。

        1、懶加載(懶漢)

        所謂懶加載,就是直到第一次被調(diào)用時(shí)才加載。其實(shí)現(xiàn)需要考慮并發(fā)問(wèn)題和指令重排,代碼如下_

        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;    }}

        這段代碼精簡(jiǎn)至極,沒(méi)有一個(gè)字符是多余的,下面逐行解讀一下_

        首先,注意到①處的volatile關(guān)鍵字,她具備兩項(xiàng)特性_

        一是保證此變量對(duì)于所有線程的可見(jiàn)性。即當(dāng)一條線程修改了這個(gè)變量的值,新值對(duì)于其他線程來(lái)說(shuō)是可以立即得知的。

        二是禁止指令重排序優(yōu)化。

        這里解釋一下指令重排序優(yōu)化_

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

          分配內(nèi)存調(diào)用構(gòu)造函數(shù)初始化成實(shí)例
        1. instance指向分配的內(nèi)存空間

        JVM 允許在保證結(jié)果正確的前提下進(jìn)行指令重排序優(yōu)化。即如上 3 步可能的順序?yàn)?->2->3 或 1->3->2 。如果順序是 1->3->2 ,當(dāng) 3 執(zhí)行完,2 還未執(zhí)行時(shí),另一個(gè)線程執(zhí)行到代碼 ③ 處,發(fā)現(xiàn)instance不為null,直接返回還未初始化好的instance并使用,就會(huì)報(bào)錯(cuò)。

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

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

        最后,當(dāng)客戶端調(diào)用Singleton.getInstance()時(shí),先檢查是否已經(jīng)實(shí)例化(代碼③),未實(shí)例化時(shí)同步代碼塊,然后再次檢查是否已實(shí)例化(代碼④),然后才執(zhí)行代碼⑤。兩次檢查的意義在于,防止synchronized同步過(guò)程中其他線程進(jìn)行了實(shí)例化。

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

        TIPS:

        網(wǎng)上也有直接對(duì)getInstance()方法加鎖的版本,這樣大范圍的方法級(jí)別加鎖會(huì)導(dǎo)致并發(fā)變低,實(shí)際上第一次調(diào)用生成實(shí)例之后,后續(xù)獲取實(shí)例根本不需要并發(fā)控制了。而本例的雙重檢查鎖版本可以避免此并發(fā)問(wèn)題。

        2、預(yù)加載(餓漢)

        與懶加載相對(duì)應(yīng),預(yù)加載是在類(lèi)加載時(shí)就已經(jīng)初始化好了,所以是天然線程安全的,代碼如下_

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

        注意到 ① 處的類(lèi)變量使用了final

        這里用final更多的意義在于提供語(yǔ)法約束。畢竟你是單例,就只有這一個(gè)實(shí)例,不可能再指向另一個(gè)。instance有了final的約束,后面再有人不小心編寫(xiě)了修改其指向的代碼就會(huì)報(bào)語(yǔ)法錯(cuò)誤。

        這就好比@Override注解,你能保證寫(xiě)對(duì)方法名和參數(shù),那不寫(xiě)注解也沒(méi)問(wèn)題,但是有了注解的約束,編譯器就會(huì)幫你檢查,還能防止別人亂改。

        3、靜態(tài)內(nèi)部類(lèi)

        此方法和預(yù)加載原理相同,都是利用JVM類(lèi)加載的特性實(shí)現(xiàn)天然的線程安全,不同之處在于,靜態(tài)內(nèi)部類(lèi)做到了延遲加載。

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

        SingletonHolder 是靜態(tài)內(nèi)部類(lèi),當(dāng)外部類(lèi)Singleton被加載的時(shí)候并不會(huì)創(chuàng)建任何實(shí)例,只有當(dāng)Singleton.getInstance()被調(diào)用的時(shí)候,才會(huì)創(chuàng)建Singleton實(shí)例,這一切由 JVM 天然完成,所以既保證了線程安全,又實(shí)現(xiàn)了延遲加載。

        4、枚舉

        沒(méi)錯(cuò),枚舉可以實(shí)現(xiàn)單例,而且這種方式是《Effective Java中文版》第二版 中的推薦實(shí)現(xiàn)方式。代碼極其簡(jiǎn)單_

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

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

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

        枚舉的構(gòu)造器總是私有的,所以不必像前幾種方式一樣顯式定義私有構(gòu)造方法
      1. 枚舉類(lèi)中的每個(gè)值,都是實(shí)例(只有INSTANCE這一個(gè)實(shí)例)

        除此之外,枚舉還附帶了一些額外好處_無(wú)償?shù)靥峁┝诵蛄谢瘷C(jī)制,還可以防止通過(guò)多次反序列化生成多個(gè)實(shí)例。

        鑒于此,單例的最佳實(shí)踐就是用枚舉來(lái)實(shí)現(xiàn)。

        5、總結(jié)

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

        所以,只要吃透了文中列出的這 4 種單例方式,就能做到舉一反三,見(jiàn)到別人寫(xiě)的單例也能一眼看出對(duì)錯(cuò)。

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

        讀取或設(shè)置一個(gè)類(lèi)型的靜態(tài)字段(被 final 修飾、已在編譯期把結(jié)果放入常量池的靜態(tài)字段除外)的時(shí)候。

        調(diào)用一個(gè)類(lèi)型的靜態(tài)方法的時(shí)候。

        懶加載,預(yù)加載和靜態(tài)內(nèi)部類(lèi)正是利用了這兩點(diǎn)特性。

        對(duì)static關(guān)鍵字遺忘的同學(xué)可以參看我的另一篇文章_《一題搞定static關(guān)鍵字》

        最后,再次強(qiáng)調(diào)一下,如果大家開(kāi)發(fā)中需要手寫(xiě)單例,建議聽(tīng)從 Joshua Bloch在《Effective Java中文版》第二版 中的建議_

        單元素的枚舉類(lèi)型已經(jīng)成為實(shí)現(xiàn) Singleton 的最佳方法

        參考資料_

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

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

        3、深入淺出單實(shí)例SINGLETON設(shè)計(jì)模式

        相關(guān)原創(chuàng)推薦

        一題搞定static關(guān)鍵字

      2.  
        (文/企資小編)
        打賞
        免責(zé)聲明
        本文為企資小編推薦作品?作者: 企資小編。歡迎轉(zhuǎn)載,轉(zhuǎn)載請(qǐng)注明原文出處:http://m.sneakeraddict.net/news/show-169394.html 。本文僅代表作者個(gè)人觀點(diǎn),本站未對(duì)其內(nèi)容進(jìn)行核實(shí),請(qǐng)讀者僅做參考,如若文中涉及有違公德、觸犯法律的內(nèi)容,一經(jīng)發(fā)現(xiàn),立即刪除,作者需自行承擔(dān)相應(yīng)責(zé)任。涉及到版權(quán)或其他問(wèn)題,請(qǐng)及時(shí)聯(lián)系我們郵件:weilaitui@qq.com。
         

        Copyright ? 2016 - 2023 - 企資網(wǎng) 48903.COM All Rights Reserved 粵公網(wǎng)安備 44030702000589號(hào)

        粵ICP備16078936號(hào)

        微信

        關(guān)注
        微信

        微信二維碼

        WAP二維碼

        客服

        聯(lián)系
        客服

        聯(lián)系客服:

        在線QQ: 303377504

        客服電話: 020-82301567

        E_mail郵箱: weilaitui@qq.com

        微信公眾號(hào): weishitui

        客服001 客服002 客服003

        工作時(shí)間:

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

        反饋

        用戶
        反饋

        中文国产成人精品久久亚洲精品AⅤ无码精品 | 成人无码区免费A∨直播| 2024你懂的网站无码内射 | 中文午夜乱理片无码| 无码国产伦一区二区三区视频| 亚洲国产综合精品中文第一| 一本精品中文字幕在线| 国产精品无码午夜福利| 天天爽亚洲中文字幕| 亚洲VA中文字幕无码一二三区 | 中文在线最新版天堂bt| 亚洲AV无码日韩AV无码导航| 亚洲乳大丰满中文字幕| 无码精品久久久天天影视| 日韩中文字幕视频| 国产成人亚洲综合无码精品| 亚洲不卡无码av中文字幕| 久久综合一区二区无码| 伊人久久精品无码av一区| 无码AV中文字幕久久专区| 国产精品无码专区| 红桃AV一区二区三区在线无码AV | 无码中文字幕乱在线观看| 亚洲伦另类中文字幕| 精品人妻无码专区中文字幕| 亚洲精品无码永久中文字幕| 精品999久久久久久中文字幕| 久久亚洲国产成人精品无码区| 无码人妻一区二区三区在线 | 亚洲AV无码久久寂寞少妇| 最近中文字幕2019视频1| 亚洲精品无码专区2| 精品无码一区在线观看| 亚洲色无码专区在线观看| 日韩免费码中文在线观看| 欧美日本道中文高清| 国产啪亚洲国产精品无码| 无码内射中文字幕岛国片| 精品亚洲AV无码一区二区| 中文字幕日本高清| 人妻少妇精品中文字幕av蜜桃|