您好,登錄后才能下訂單哦!
這篇文章主要為大家展示了“List集合多線程并發條件下不安全怎么辦”,內容簡而易懂,條理清晰,希望能夠幫助大家解決疑惑,下面讓小編帶領大家一起研究并學習一下“List集合多線程并發條件下不安全怎么辦”這篇文章吧。
在日常開發過程中,List是我們常用的集合,比如查詢數據庫內容返回值比會用一個集合來裝,但是在多線程并發的條件下,會出現安全問題嗎?下面我們就來測試一下,如果出現安全問題,該如何解決.
public static void main(String[] args) { // List集合 List<String> list = new ArrayList<>(); // 循環插入 for (int i = 0; i < 10; i++) { list.add(UUID.randomUUID().toString().substring(0,5)); System.out.println(list); } }
可以看到單線程條件下,我們做list的插入操作完全沒問題,下面我們來模擬并發條件下執行,會出現什么問題。
public static void main(String[] args) { // List集合 List<String> list = new ArrayList<>(); // 循環插入 for (int i = 0; i < 10; i++) { // 開啟線程執行 new Thread(()->{ list.add(UUID.randomUUID().toString().substring(0,5)); System.out.println(list); },"線程List").start(); } }
ArrayList在迭代的時候如果同時對其進行修改就會拋出java.util.ConcurrentModificationException異常,就是并發修改異常。
public static void main(String[] args) { // List集合 List<String> list = new Vector<>(); // 循環插入 for (int i = 0; i < 10; i++) { // 開啟線程執行 new Thread(()->{ list.add(UUID.randomUUID().toString().substring(0,5)); System.out.println(list); },"線程List").start(); } }
Vector 是同步訪問的,它的add方法底層加了synchronized關鍵字修飾。
測試結果:
public static void main(String[] args) { // List集合 List<String> list = Collections.synchronizedList(new ArrayList<>()); // 循環插入 for (int i = 0; i < 10; i++) { // 開啟線程執行 new Thread(()->{ list.add(UUID.randomUUID().toString().substring(0,5)); System.out.println(list); },"線程List").start(); } }
查看底層源碼可以發現他也使用了synchronized關鍵字修飾。
public static void main(String[] args) { // List集合 List<String> list = new CopyOnWriteArrayList<>(); // 循環插入 for (int i = 0; i < 10; i++) { // 開啟線程執行 new Thread(()->{ list.add(UUID.randomUUID().toString().substring(0,5)); System.out.println(list); },"線程List").start(); } }
查看源碼它使用的是lock鎖機制。
寫入時復制,有多個線程調用的時候,寫入的時候,復制一份,避免覆蓋造成數據問題。就是在寫的時候不對原集合進行修改,而是重新復制一份,修改完之后,再移動指針。
從JDK1.5開始Java并發包里提供了兩個使用CopyOnWrite機制實現的并發容器,它們是CopyOnWriteArrayList和CopyOnWriteArraySet。CopyOnWrite容器非常有用,可以在非常多的并發場景中使用到。
解讀源碼:
/** * Appends the specified element to the end of this list. * * @param e element to be appended to this list * @return {@code true} (as specified by {@link Collection#add}) */ public boolean add(E e) { final ReentrantLock lock = this.lock;//可重入鎖 lock.lock();//加鎖 try { Object[] elements = getArray(); int len = elements.length; Object[] newElements = Arrays.copyOf(elements, len + 1);//拷貝新數組 newElements[len] = e; setArray(newElements);//將引用指向新數組 return true; } finally { lock.unlock();//解鎖 } }
add()在添加集合的時候加上了鎖,保證了同步,避免了多線程寫的時候會Copy出N個副本出來。
以上是“List集合多線程并發條件下不安全怎么辦”這篇文章的所有內容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內容對大家有所幫助,如果還想學習更多知識,歡迎關注億速云行業資訊頻道!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。