深圳罗湖区网站搜狗搜索推广
ArrayList这个集合在实际开发中我们经常用到,但是在多线程中我们是不可以使用的,这也正是因为他是一个线程不安全的容器,同时还有HashMap(看下一篇文章),这里我们先说一下ArrayList。
解决方法:
改进的方法很简单,如程序上的注释行一样,使用线程安全的 Vector 代替 ArrayList 即可。
package com.zmkj.admin.test;import java.util.ArrayList;/*** 线程不安全的容器:ArrayList** ArrayList 在多线程中,是不安全的,可能会导致程序出错。* 这里我们探究一下 为什么会出错* @author sunminghao*/
public class ArrayListMultiThread {static ArrayList<Integer> list = new ArrayList<Integer>(10);
// static Vector<Integer> list = new Vector<Integer>(10);public static class AddThread implements Runnable {@Overridepublic void run() {for (int i = 0; i < 1000000; i++) {list.add(i);}}}public static void main(String[] args) throws InterruptedException {Thread t1 = new Thread(new AddThread());Thread t2 = new Thread(new AddThread());t1.start();t2.start();t1.join();t2.join();System.out.println(list.size());}/*** 在执行过程中 其实可能会出现三种结果** 一、程序正常结束,ArrayList 的最终结果确实是200W 。这说明即使并行程序有问题,也未必会每次都变现出来。* 二、程序抛出异常:* Exception in thread "Thread-1" java.lang.ArrayIndexOutOfBoundsException: 4164* at java.util.ArrayList.add(ArrayList.java:463)* at com.zmkj.admin.test.ArrayListMultiThread$AddThread.run(ArrayListMultiThread.java:20)* at java.lang.Thread.run(Thread.java:748)* 1001008** 这是因为ArrayList 在扩容过程中,内部一致性被破坏,但由于没有锁的保护,另外一个线程访问到了不一致的内部状态,导致出现越界的问题。** 三、出现了一个非常隐蔽的错误,比如打印如下值作为ArrayList 的大小。输出如下:* 1793758** 这是由于线程访问冲突,是的保存容器大小的变凉被多线程不正常的访问,同时两个线程也对ArrayList 中的同一个位置进行赋值导致的。* 如果出现这种问题,那么很不幸,你就的得到了一个没有错误提示的错误。并且也是未必可以复现的。*** TODO 注意: 改进的方法很简单,如程序上的注释行一样,使用线程安全的 Vector 代替 ArrayList 即可。*/
}