多线程---线程安全的集合类

文章目录

我们原先学习过的集合类,大部分都是线程不安全的。只有Vector、Stack、HashTable是线程安全的,但是也不推荐使用。通常情况下,我们都会使用下面介绍的这几种数据结构来保证多线程中的线程安全。

多线程环境使用数组

synchronizedList

synchronizedList是通过对list的每个操作都加上synchronized来保证线程安全的。

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

CopyOnWriteArrayList

CopyOnwrite:写时拷贝。它不加锁就能保证线程安全,采用"双缓冲区机制"。但是适用的场景有限:通常用在写的频率比较低的场景,如一写多读。

注: 当数据量非常大的时候,拷贝数据的消耗非常大,就不适合使用这样的方式了。

多线程环境使用队列

在多线程环境下我们都是借助阻塞队列来保证线程安全,但是又可以分为以下几种类型:

java 复制代码
        //普通的阻塞队列
		BlockingQueue<Integer> blockingQueue = new LinkedBlockingDeque<>();
		//基于数组实现的阻塞队列
        ArrayBlockingQueue<Integer> arrayBlockingQueue = new ArrayBlockingQueue<Integer>(10);
        //基于链表实现的阻塞队列
        LinkedBlockingDeque<Integer> linkedBlockingDeque = new LinkedBlockingDeque<>();
        //带有优先级的阻塞队列
        PriorityBlockingQueue<Integer> priorityBlockingQueue = new PriorityBlockingQueue<>();

        //最多只包含一个元素的阻塞队列
        TransferQueue<Integer> transferQueue = new LinkedTransferQueue<>();

多线程环境使用哈希表

HashTable

不推荐使用。它虽然能够保证线程安全,但是使用的方法是:无脑的给每个方法都加上synchronized。代码的执行效率并不高。

ConcurrentHashMap

ConcurrentHashMap背后做了非常多的优化使其在多线程环境下非常好用

  1. 优化锁粒度的控制
  • HashTable直接在方法上加锁,相当于是对this加锁。所有对HashTable的操作都会造成锁竞争,锁冲突的概率很大。
  • ConcurrentHashMap是分别对每个哈希桶加锁,即加了多把锁,这个就极大程度的减小了锁冲突的概率。
  1. ConcurrentHashMap只对写操作加锁,不对读操作加锁

    它这样做的时候会有以下几种情况:

    • 两个线程同时读---没有锁冲突
    • 两个线程同时写---有锁冲突
    • 一个线程读、一个线程写呢?---也没有锁冲突
      这样会不会线程不安全呢?即读到一个修改了一半的值。其实是不会的,ConcurrentHashMap在设计的时候就保证了读到的数据一定是一个完整的数据(要么是旧版本,要么是新版本);另外它还会广泛的使用到volatile来保证内存可见性,进而保证线程安全。
  2. 充分利用到了CAS特性

  • ConcurrentHashMap的思路就是能不加锁就不加锁,加锁毕竟是有开销的。它就广泛的应用了CAS的特性,比如维护元素个数。在加锁时也会使用基于CAS实现的轻量级锁。
  1. 优化了ConcurrentHashMap的扩容操作


HashTable VS HashMap VS ConcurrentHashMap

  1. HashMap是线程不安全的,HashTable和ConcurrentHashMap是线程安全的。
  2. HashMap的key值可以为null,HashTable和ConcurrentHashMap的key值不能为null
  3. ConcurrentHashMap相对于HashTable的四大优化
相关推荐
风生u4 分钟前
activiti7 详解
java
岁岁种桃花儿12 分钟前
SpringCloud从入门到上天:Nacos做微服务注册中心(二)
java·spring cloud·微服务
Word码16 分钟前
[C++语法] 继承 (用法详解)
java·jvm·c++
TT哇22 分钟前
【实习 】银行经理端两个核心功能的开发与修复(银行经理绑定逻辑修复和线下领取扫码功能开发)
java·vue.js
逝水如流年轻往返染尘24 分钟前
Java中的数组
java
java1234_小锋41 分钟前
Java高频面试题:BIO、NIO、AIO有什么区别?
java·面试·nio
用户8307196840821 小时前
Java IO三大模型(BIO/NIO/AIO)超详细总结
java
sheji34161 小时前
【开题答辩全过程】以 基于SSM的花店销售管理系统为例,包含答辩的问题和答案
java
Mr_sun.1 小时前
Day09——入退管理-入住-2
android·java·开发语言
MAGICIAN...1 小时前
【java-软件设计原则】
java·开发语言