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

寫了個全局變量的bug,被同事們打臉

安全
話說棧長前陣子寫了一個功能,測試 0 bug 就上線了,上線后也運行好好的,好多天都沒有人反饋bug,超爽。。不出問題還好,出問題就是大問題。。

話說棧長前陣子寫了一個功能,測試 0 bug 就上線了,上線后也運行好好的,好多天都沒有人反饋bug,超爽。。

不出問題還好,出問題就是大問題。。

最近有個客戶反饋某些數據混亂問題,看代碼死活看不出什么問題,很詭異,再仔細看代碼,原來是一個全局變量的問題,導致在并發情況下出現了線程不安全的問題,事后被同事們打臉!!!

慎用全局變量,我在公司一直在強調,沒想到這么低級的問題居然發生在自己身上,說起來真的慚愧啊。。

最開始使用的是 Spring 注入對象的方式: 

  1. @Autowired 
  2. private Object object; 

因為 Spring 默認是單例,所以這樣寫是沒有問題的,后來隨著業務的發展,需要多個不同的業務實例,我改成了這種方式: 

  1. @Setter 
  2. private Object object; 

這個 @Setter 是 Lombok 的注解,用來生成 setters 方法,現在想起來,真是低級啊,同時操作的情況下,這個對象肯定會出現覆蓋的情況,從而導致上面說的問題。

寫了一個這么低級bug,我也不怕不好意思發出來,大家都謹記一下吧。

另外,我再總結幾個慎用全局變量的場景:

1、SimpleDateFormat

SimpleDateFormat 禁止定義成 static 變量或者全局共享變量,因為它是線程不安全的,都被寫進阿里巴巴的《Java開發手冊》里了: 

寫了個全局變量的bug,被同事們打臉

為什么說 SimpleDateFormat 不是線程安全的呢?

來看下它的 format 方法源碼: 

寫了個全局變量的bug,被同事們打臉

可以看到 calendar 變量居然也是全局變量,多線程情況下就會存在設置臟變量的情況。

所以,如果要用 SimpleDateFormat,就在每次用的時候都創建一個 SimpleDateFormat 對象,做到線程間隔離。

2、資源連接

資源連接包括數據庫連接、FTP連接、Redis連接等,這種也要慎用全局變量,一量使用全局變量,就會遇到以下問題:

  • 關閉連接的時候,就可能把別人正在操作的連接給關了,導致其他線程的業務中斷;
  • 因為是全局變量,創建的時候可能會創建多個實例,在關閉連接的時候,就可能只關閉了一個對象的連接,造成其他連接沒有被關閉,最后導致連接耗光系統不可用;

3、數字運算

這也是個很經典的問題了,如果要用多線程對一個數字進行累加等其他運算處理,千萬不要用全局基礎類型的變量,如下所示:

  1. private long count

多線程情況下,某個線程獲取到的值可能已經被其他線程修改了,最后得到的值就不準確了。

當然,上面的示例可以通過加鎖的方式來解決,也可以使用全局的原子類(java.util.concurrent.atomic.Atom*)進行處理,比如:

  1. private AtomicInteger count = new AtomicInteger(); 

注意,這種原子類使用全局變量就沒有線程安全的問題,它使用了 CAS 算法保證了數據一致性。

不過,阿里推薦使用LongAdder,因為性能更好:

  1. java.util.concurrent.atomic.LongAdder 

 寫了個全局變量的bug,被同事們打臉

4、全局session

來看下面的例子: 

  1. @Autowired 
  2. protected HttpSession session; 

全局注入一個 Session 對象,在 Spring 中,這樣全局注入使用上面是默認沒問題的,包括 request, response 對象,都可以通過全局注入來獲取。

這樣會存在線程安全性嗎?

不會!

使用這種方式,當 Bean 初始化時,Spring 并沒有注入真實對象,而是注入了一個代理對象,真正使用的時候通過該代理對象獲取真正的對象。

并且,在注入此類對象時,Spring使用了線程局部變量(ThreadLocal),這就保證了 request/response/session 對象的線程安全性了。

具體就不展開了,詳細的介紹及測試大家可以點擊這個鏈接查看這篇文章。

既然是線程安全,但也得小心,如果我在方法中主動使 session 對象失效并重建了: 

  1. session.invalidate(); 
  2. session = request.getSession(); 

這樣,session對象就變成了真實對象了,不再是代理對象,就變成了文章最開始的時候我說的那種多線程安全問題了,如果線上出現 session 會話混亂,用戶 A 就可能看到用戶 B 的數據,你想想可不可怕?

所以,即使可以這樣使用,也得千萬小心謹慎,最好是在方法級別使用這些對象。

總結

今天,棧長總結了一下我是怎么寫出這個全局變量的低級 bug,也總結了下慎用全局變量的 4 種情況,相信大家多多少都遇到過類似的問題,希望能幫助大家少踩坑。

全局變量雖好,但我們也得謹慎使用啊,一定要考慮是否引起多線程安全問題,不然會引起重大問題。

 

責任編輯:未麗燕 來源: 今日頭條
相關推薦

2024-05-27 09:18:18

2013-07-17 16:16:06

Android全局變量定義全局變量Application

2010-03-09 14:12:55

Python全局變量

2025-06-03 08:50:00

Static全局變量C 語言

2010-11-12 10:08:55

SQL Server全

2013-07-25 15:15:26

iOS開發學習iOS全局變量

2010-02-01 14:28:37

Python全局變量

2009-09-24 09:28:00

Scala講座全局變量scala

2009-11-06 13:28:19

Javascript框

2013-07-22 14:07:47

2024-05-29 08:49:22

Python全局變量局部變量

2015-01-07 14:41:32

Android全局變量局部變量

2021-05-27 08:59:09

Go匯編命令

2017-02-08 12:28:37

Android變量總結

2009-12-09 17:07:08

PHP unset全局

2014-06-23 10:25:12

2010-09-08 17:25:17

SQL全局變量

2009-12-04 13:14:19

PHP Global變

2022-05-23 07:56:58

C語言嵌入式開發

2009-12-04 13:31:21

PHP全局變量不能生效
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 国产成人精品一区二区三区四区 | 欧美自拍一区 | 翔田千里一区二区 | 九九九久久国产免费 | 91在线视频精品 | 久久久久久久久99 | 久久久精品一区 | 在线视频一区二区三区 | 日韩免费一区 | 97视频在线观看免费 | 日韩精品久久一区 | 成人二区 | 99视频在线 | 国产精品一区一区三区 | 国产xxx在线观看 | 欧美日韩成人一区二区 | 日韩一级| 视频羞羞 | 岛国av一区二区 | 在线观看你懂的网站 | 不卡欧美 | 欧美一区二区三区高清视频 | 国产精品一区二区久久 | 欧美一区二区三区视频在线观看 | 天天夜夜操 | 一级毛片色一级 | 免费看a | 2018国产精品 | 精品产国自在拍 | 另类专区成人 | 免费观看www7722午夜电影 | 成人在线不卡 | 狠狠干2020 | 久久y| 久久丁香 | 在线观看视频福利 | 香蕉婷婷 | 欧美一区 | 91在线视频免费观看 | 久久精品久久久久久 | 国产精品毛片在线 |