【JUC】三、集合的线程安全

文章目录

1、ArrayList集合线程安全问题分析

对List集合非线程安全的Demo代码:

java 复制代码
public class ArrayListDemo {

    public static void main(String[] args) {
        List<String> list = new ArrayList<>();
        //多个线程同时写入List集合
        for (int i = 0; i < 10; i++) {
            new Thread(() -> {
                //加元素
                list.add(UUID.randomUUID().toString().substring(0,8));
                //遍历输出集合
                System.out.println(list);
            },String.valueOf(i)).start();
        }
    }
}

运行:

ConcurrentModificationException异常,是在多线程环境下,当一个线程正在遍历集合,而另一个线程对集合进行了修改操作时,就会抛出这个异常。以ArrayList为例,其add方法源码,未加synchronized关键字:

再点击报错详情,进入抛出异常的方法:

modCount即集合新增的次数,是实际修改次数,而expectedModCount是预期修改次数,它是ArrayList的一个内部类Itr的成员变量,调用iterator()获取迭代器时,内部创建Itr对象,此时,modCount会赋值给expectedModCount:

拿到迭代器对象,要遍历集合时,modCount已经赋值给expectedModCount,而此时其他线程继续add,modCount+1,modCount和expectedModCount就不相等了。

2、解决方式一:Vector或synchronizedList( )

List接口的另一个实现类Vector,其add方法加了关键字,使用它可解决线程安全问题,但很古老了,since1.2,很少用了。

java 复制代码
List<String> list = new Vector<>();
//重复代码略

同样一种古老的解决方案,可以用Collections的synchronizedList方法,传入一个有线程安全问题的List,如ArrayList:

java 复制代码
List<String> list = Collections.synchronizedList(new ArrayList<>());

3、解决方式二:CopyOnWriteArrayList 写时复制

java 复制代码
List<String> list = new CopyOnWriteArrayList<>();

完整demo:

java 复制代码
public class ArrayListDemo {

    public static void main(String[] args) {
        List<String> list = new CopyOnWriteArrayList<>();
        //多个线程同时写入List集合
        for (int i = 0; i < 10; i++) {
            new Thread(() -> {
                //加元素
                list.add(UUID.randomUUID().toString().substring(0,8));
                //遍历输出集合
                System.out.println(list);
            },String.valueOf(i)).start();
        }
    }
}

原理是写时复制技术,即:

  • 对这个List实现类的集合,可以多线程并发读
  • 往集合中写的时候,则只能独立写,先复制一份原来的集合,这个时候读还是读原来的,然后往新集合里面写入新的内容
  • 写完后新旧合并,再读时,就读这个合并后的集合

看下源码,再对照着理解写时复制:

4、HashSet集合线程不安全的分析与解决

java 复制代码
public class HashSetDemo {

    public static void main(String[] args) {
        Set<String> set = new HashSet<>();
        for (int i = 0; i < 30; i++) {
            new Thread(() -> {
                //写入
                set.add(UUID.randomUUID().toString().substring(0,8));
                //读
                System.out.println(set);
            },String.valueOf(i)).start();
        }
    }
}

运行:

解决办法类比上面的List,使用CopyOnWriteArraySet

java 复制代码
Set<String> set = new CopyOnWriteArraySet<>();

5、HashMap集合线程不安全的分析与解决

java 复制代码
public class HashSetDemo {

    public static void main(String[] args) {
        Map<String,string> map = new HashMap<>();
        for (int i = 0; i < 30; i++) {
        	String key = String.valueOf(i);
            new Thread(() -> {
                //写入
                map.put(key,UUID.randomUUID().toString().substring(0,8));
                //读
                System.out.println(map);
            },String.valueOf(i)).start();
        }
    }
}

解决办法类比List,用ConcurrentHashMap

java 复制代码
Map<String,String> map = new ConcurrentHashMap<>();
相关推荐
kkkkkkkkl243 天前
Spring Boot 中基于线程池的订单创建并行化实践
java·spring boot·juc
_OP_CHEN3 天前
【C++数据结构进阶】吃透 LRU Cache缓存算法:O (1) 效率缓存设计全解析
数据结构·数据库·c++·缓存·线程安全·内存优化·lru
ZHang......5 天前
synchronized(三)
开发语言·笔记·juc
tryxr5 天前
线程安全的类 ≠ 线程安全的程序
java·开发语言·vector·线程安全
Jack_abu7 天前
详解java中的线程间协作工具:CountDownLatch,Semaphore,CyclicBarrier(二)
java·并发编程·juc
Jack_abu7 天前
详解java中的BlockingQueue阻塞队列
java·juc·阻塞队列·blockingqueue
代码游侠10 天前
学习笔记——写时复制(Copy-on-Write)
linux·网络·笔记·学习·写时复制
努力发光的程序员10 天前
互联网大厂Java求职面试实录
java·jvm·线程池·多线程·hashmap·juc·arraylist
Brookty11 天前
Java并发编程核心的基础知识
java·开发语言·java-ee·多线程·线程安全
BestOrNothing_201521 天前
C++ 并发四件套:并发编程 / 原子性 / 数据竞争 / 内存模型 (全解析)
c++·多线程·并发编程·线程安全·内存模型·原子操作·数据竞争