• 當前位置:聯(lián)升科技 > 技術(shù)資訊 > 開(kāi)發(fā)技術(shù) >

    寫(xiě)了個(gè)全局變量的bug,被同事們打臉

    2020-11-09    作者:Java技術(shù)棧    來(lái)源:今日頭條    閱讀: 次
    話(huà)說(shuō)棧長(cháng)前陣子寫(xiě)了一個(gè)功能,測試 0 bug 就上線(xiàn)了,上線(xiàn)后也運行好好的,好多天都沒(méi)有人反饋bug,超爽。。
    不出問(wèn)題還好,出問(wèn)題就是大問(wèn)題。。
    最近有個(gè)客戶(hù)反饋某些數據混亂問(wèn)題,看代碼死活看不出什么問(wèn)題,很詭異,再仔細看代碼,原來(lái)是一個(gè)全局變量的問(wèn)題,導致在并發(fā)情況下出現了線(xiàn)程不安全的問(wèn)題,事后被同事們打臉!!!
    慎用全局變量,我在公司一直在強調,沒(méi)想到這么低級的問(wèn)題居然發(fā)生在自己身上,說(shuō)起來(lái)真的慚愧啊。。
    最開(kāi)始使用的是 Spring 注入對象的方式:
    @Autowired 
    private Object object; 
    因為 Spring 默認是單例,所以這樣寫(xiě)是沒(méi)有問(wèn)題的,后來(lái)隨著(zhù)業(yè)務(wù)的發(fā)展,需要多個(gè)不同的業(yè)務(wù)實(shí)例,我改成了這種方式:
    @Setter 
    private Object object; 
    這個(gè) @Setter 是 Lombok 的注解,用來(lái)生成 setters 方法,現在想起來(lái),真是低級啊,同時(shí)操作的情況下,這個(gè)對象肯定會(huì )出現覆蓋的情況,從而導致上面說(shuō)的問(wèn)題。
    寫(xiě)了一個(gè)這么低級bug,我也不怕不好意思發(fā)出來(lái),大家都謹記一下吧。
    另外,我再總結幾個(gè)慎用全局變量的場(chǎng)景:
    1、SimpleDateFormat
    SimpleDateFormat 禁止定義成 static 變量或者全局共享變量,因為它是線(xiàn)程不安全的,都被寫(xiě)進(jìn)阿里巴巴的《Java開(kāi)發(fā)手冊》里了:

    寫(xiě)了個(gè)全局變量的bug,被同事們打臉
    最新的完整版可以點(diǎn)擊文末下面的了解更多鏈接關(guān)注:Java技術(shù)棧,回復 “手冊” 獲取。
    為什么說(shuō) SimpleDateFormat 不是線(xiàn)程安全的呢?
    來(lái)看下它的 format 方法源碼:
    寫(xiě)了個(gè)全局變量的bug,被同事們打臉

    可以看到 calendar 變量居然也是全局變量,多線(xiàn)程情況下就會(huì )存在設置臟變量的情況。
    所以,如果要用 SimpleDateFormat,就在每次用的時(shí)候都創(chuàng )建一個(gè) SimpleDateFormat 對象,做到線(xiàn)程間隔離。
    2、資源連接
    資源連接包括數據庫連接、FTP連接、Redis連接等,這種也要慎用全局變量,一量使用全局變量,就會(huì )遇到以下問(wèn)題:
    關(guān)閉連接的時(shí)候,就可能把別人正在操作的連接給關(guān)了,導致其他線(xiàn)程的業(yè)務(wù)中斷;
    因為是全局變量,創(chuàng )建的時(shí)候可能會(huì )創(chuàng )建多個(gè)實(shí)例,在關(guān)閉連接的時(shí)候,就可能只關(guān)閉了一個(gè)對象的連接,造成其他連接沒(méi)有被關(guān)閉,最后導致連接耗光系統不可用;
    3、數字運算
    這也是個(gè)很經(jīng)典的問(wèn)題了,如果要用多線(xiàn)程對一個(gè)數字進(jìn)行累加等其他運算處理,千萬(wàn)不要用全局基礎類(lèi)型的變量,如下所示:
    private long count; 
    多線(xiàn)程情況下,某個(gè)線(xiàn)程獲取到的值可能已經(jīng)被其他線(xiàn)程修改了,最后得到的值就不準確了。
    當然,上面的示例可以通過(guò)加鎖的方式來(lái)解決,也可以使用全局的原子類(lèi)(java.util.concurrent.atomic.Atom*)進(jìn)行處理,比如:
    private AtomicInteger count = new AtomicInteger(); 
    注意,這種原子類(lèi)使用全局變量就沒(méi)有線(xiàn)程安全的問(wèn)題,它使用了 CAS 算法保證了數據一致性。
    不過(guò),阿里推薦使用LongAdder,因為性能更好:
    java.util.concurrent.atomic.LongAdder 
    寫(xiě)了個(gè)全局變量的bug,被同事們打臉

    4、全局session
    來(lái)看下面的例子:
    @Autowired 
    protected HttpSession session; 
    全局注入一個(gè) Session 對象,在 Spring 中,這樣全局注入使用上面是默認沒(méi)問(wèn)題的,包括 request, response 對象,都可以通過(guò)全局注入來(lái)獲取。
    這樣會(huì )存在線(xiàn)程安全性嗎?
    不會(huì )!
    使用這種方式,當 Bean 初始化時(shí),Spring 并沒(méi)有注入真實(shí)對象,而是注入了一個(gè)代理對象,真正使用的時(shí)候通過(guò)該代理對象獲取真正的對象。
    并且,在注入此類(lèi)對象時(shí),Spring使用了線(xiàn)程局部變量(ThreadLocal),這就保證了 request/response/session 對象的線(xiàn)程安全性了。
    具體就不展開(kāi)了,詳細的介紹及測試大家可以點(diǎn)擊這個(gè)鏈接查看這篇文章。
    既然是線(xiàn)程安全,但也得小心,如果我在方法中主動(dòng)使 session 對象失效并重建了:
    session.invalidate(); 
    session = request.getSession(); 
    這樣,session對象就變成了真實(shí)對象了,不再是代理對象,就變成了文章最開(kāi)始的時(shí)候我說(shuō)的那種多線(xiàn)程安全問(wèn)題了,如果線(xiàn)上出現 session 會(huì )話(huà)混亂,用戶(hù) A 就可能看到用戶(hù) B 的數據,你想想可不可怕?
    所以,即使可以這樣使用,也得千萬(wàn)小心謹慎,最好是在方法級別使用這些對象。
    總結
    今天,棧長(cháng)總結了一下我是怎么寫(xiě)出這個(gè)全局變量的低級 bug,也總結了下慎用全局變量的 4 種情況,相信大家多多少都遇到過(guò)類(lèi)似的問(wèn)題,希望能幫助大家少踩坑。
    全局變量雖好,但我們也得謹慎使用啊,一定要考慮是否引起多線(xiàn)程安全問(wèn)題,不然會(huì )引起重大問(wèn)題。


    相關(guān)文章

    我們很樂(lè )意傾聽(tīng)您的聲音!
    即刻與我們取得聯(lián)絡(luò )
    成為日后肩并肩合作的伙伴。

    行業(yè)資訊

    聯(lián)系我們

    13387904606

    地址:新余市仙女湖區仙女湖大道萬(wàn)商紅A2棟

    手機:13755589003
    QQ:122322500
    微信號:13755589003

    江西新余網(wǎng)站設計_小程序制作_OA系統開(kāi)發(fā)_企業(yè)ERP管理系統_app開(kāi)發(fā)-新余聯(lián)升網(wǎng)絡(luò )科技有限公司 贛ICP備19013599號-1   贛公網(wǎng)安備 36050202000267號   

    微信二維碼
    色噜噜狠狠一区二区三区果冻|欧美亚洲日本国产一区|国产精品无码在线观看|午夜视频在线观看一区|日韩少妇一区二区无码|伊人亚洲日韩欧美一区二区|国产在线码观看清码视频