話說堆疊長前陣子寫了一個功能,測驗 0 bug 就上線了,上線后也運行好好的,好多天都沒有人反饋bug,超爽,,
不出問題還好,出問題就是大問題,,
最近有個客戶反饋某些資料混亂問題,看代碼死活看不出什么問題,很詭異,再仔細看代碼,原來是一個全域變數的問題,導致在并發情況下出現了執行緒不安全的問題,事后被同事們打臉!!!
慎用全域變數,我在公司一直在強調,沒想到這么低級的問題居然發生在自己身上,說起來真的慚愧啊,,
最開始使用的是 Spring 注入物件的方式:
@Autowired
private Object object;
因為 Spring 默認是單例,所以這樣寫是沒有問題的,后來隨著業務的發展,需要多個不同的業務實體,我改成了這種方式:
@Setter
private Object object;
這個 @Setter 是 Lombok 的注解,用來生成 setters 方法,現在想起來,真是低級啊,同時操作的情況下,這個物件肯定會出現覆寫的情況,從而導致上面說的問題,
寫了一個這么低級bug,我也不怕不好意思發出來,大家都謹記一下吧,
另外,我再總結幾個慎用全域變數的場景:
1、SimpleDateFormat
SimpleDateFormat 禁止定義成 static 變數或者全域共享變數,因為它是執行緒不安全的,都被寫進阿里巴巴的《Java開發手冊》里了:

為什么說 SimpleDateFormat 不是執行緒安全的呢?
來看下它的 format 方法原始碼:

可以看到 calendar 變數居然也是全域變數,多執行緒情況下就會存在設定臟變數的情況,
所以,如果要用 SimpleDateFormat,就在每次用的時候都創建一個 SimpleDateFormat 物件,做到執行緒間隔離,
2、資源連接
資源連接包括資料庫連接、FTP連接、Redis連接等,這種也要慎用全域變數,一量使用全域變數,就會遇到以下問題:
1)關閉連接的時候,就可能把別人正在操作的連接給關了,導致其他執行緒的業務中斷;
2)因為是全域變數,創建的時候可能會創建多個實體,在關閉連接的時候,就可能只關閉了一個物件的連接,造成其他連接沒有被關閉,最后導致連接耗光系統不可用;
3、數字運算
這也是個很經典的問題了,如果要用多執行緒對一個數字進行累加等其他運算處理,千萬不要用全域基礎型別的變數,如下所示:
private long count;
多執行緒情況下,某個執行緒獲取到的值可能已經被其他執行緒修改了,最后得到的值就不準確了,
當然,上面的示例可以通過加鎖的方式來解決,也可以使用全域的原子類(java.util.concurrent.atomic.Atom*)進行處理,比如:
private AtomicInteger count = new AtomicInteger();
注意,這種原子類使用全域變數就沒有執行緒安全的問題,它使用了 CAS 演算法保證了資料一致性,
不過,阿里推薦使用LongAdder,因為性能更好:
java.util.concurrent.atomic.LongAdder

4、全域session
來看下面的例子:
@Autowired
protected HttpSession session;
全域注入一個 Session 物件,在 Spring 中,這樣全域注入使用上面是默認沒問題的,包括 request, response 物件,都可以通過全域注入來獲取,
這樣會存在執行緒安全性嗎?
不會!
使用這種方式,當 Bean 初始化時,Spring 并沒有注入真實物件,而是注入了一個代理物件,真正使用的時候通過該代理物件獲取真正的物件,
并且,在注入此類物件時,Spring使用了執行緒區域變數(ThreadLocal),這就保證了 request/response/session 物件的執行緒安全性了,
具體就不展開了,詳細的介紹及測驗大家可以點擊這個鏈接查看這篇文章,
既然是執行緒安全,但也得小心,如果我在方法中主動使 session 物件失效并重建了:
session.invalidate();
session = request.getSession();
這樣,session物件就變成了真實物件了,不再是代理物件,就變成了文章最開始的時候我說的那種多執行緒安全問題了,如果線上出現 session 會話混亂,用戶 A 就可能看到用戶 B 的資料,你想想可不可怕?
所以,即使可以這樣使用,也得千萬小心謹慎,最好是在方法級別使用這些物件,
總結
今天,堆疊長總結了一下我是怎么寫出這個全域變數的低級 bug,也總結了下慎用全域變數的 4 種情況,相信大家多多少都遇到過類似的問題,希望能幫助大家少踩坑,
全域變數雖好,但我們也得謹慎使用啊,一定要考慮是否引起多執行緒安全問題,不然會引起重大問題,
你還遇到過哪些全域變數的問題,歡迎留言分享哦!
推薦去我的博客閱讀更多:
1.Java JVM、集合、多執行緒、新特性系列教程
2.Spring MVC、Spring Boot、Spring Cloud 系列教程
3.Maven、Git、Eclipse、Intellij IDEA 系列工具教程
4.Java、后端、架構、阿里巴巴等大廠最新面試題
覺得不錯,別忘了點贊+轉發哦!
轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/164441.html
標籤:Java
