下面這個問題是 Java 程序員面試經常會遇到的吧。
工作一兩年的應該都知道 ArrayList 是線程不安全的,要使用線程安全的就使用 Vector,這也是各種 Java 面試寶典裡面所提及的,可能很多工作好幾年的程序員都停留在這個知識面上。
先說說為什麼 ArrayList 是線程不安全的吧,來看以下的代碼。
![出場率比較高的一道多線程安全面試題](http://p2.ttnews.xyz/loading.gif)
這是它的輸出結果,我們期望的結果應該都是:30000,然後並不是,這就是傳說中的多線程併發問題了。
![出場率比較高的一道多線程安全面試題](http://p2.ttnews.xyz/loading.gif)
現象分析
從以上結果可以總結出 ArrayList 在併發情況下會出現的幾種現象。
1、發生 ArrayIndexOutOfBoundsException 異常;
定位到異常所在源代碼,毫無疑問,問題是出現在多線程併發訪問下,由於沒有同步鎖的保護,造成了 ArrayList 擴容不一致的問題。
2、程序正常運行,輸出了少於實際容量的大小;
這個也是多線程併發賦值時,對同一個數組索引位置進行了賦值,所以出現少於預期大小的情況。
3、程序正常運行,輸出了預期容量的大小;
這是正常運行結果,未發生多線程安全問題,但這是不確定性的,不是每次都會達到正常預期的。
解決方案
既然這樣,那麼在高併發情況下,使用什麼樣的列表集合保護線程安全呢?回到文章最開始的地方,使用 Vector,還有別的嗎?當然有,篇幅有限,請各位看官期待後續文章。
另外,像 HashMap, HashSet 等都有類似多線程安全問題,在多線程併發環境下避免使用這種集合。
閱讀更多 java進階架構師 的文章