线程安全的集合类(ConcurrentHashMap面试超高频考点)

简介: 线程安全的集合类(ConcurrentHashMap面试超高频考点)

🧵多线程环境下使用ArrayList

✨自己使用同步机制synchronized或者ReentrantLock

🎉使用Collections.synchronizedList(new ArrayList); synchronizedList是标准库提供的一个基于synchronized进行线程同步的List,synchronizedList的关键操作上都带有synchronized来满足线程安全

🎊使用CopyOnWriteArrayList CopyOnWrite容器即写时复制容器: 当往一个容器中添加元素的时候,不往该容器中直接添加元素,而是先将该容器进行复制一个新的容器,添加的时候往复制的容器中添加 添加完后,将原容器中的引用指向新复制的容器 CopyOnWrite容器是一种读和写分离的思想,读和写的容器是不同的

优点:在读多写少的情况下,性能高

缺点:占有内存多,新写的数据不能被第一时间读到

🎡多线程环境下使用队列(Queue)

使用阻塞队列

🥎ArrayBlockingQueue,基于数组实现的阻塞队列

🏀LinkedBlockingQueue,基于链表实现的阻塞队列

🏐PriorityBlockingQueue,基于堆实现的带优先级的阻塞队列

⚽TransferQueue,最多只包含一个元素的阻塞队列

🎑多线程环境下使用哈希表(面试超高频考点)

HashMap是线程不安全的,在多线程下使用线程安全的哈希表可以使用:

🎺HashTable
🪗ConcurrentHashMap

🏅HashTable

HashTable的底层实现只是对关键方法加上synchronized

HahTable的底层数据结构为数组+链表

image.png

这样加锁相当于对HashTable对象本身加锁即对整个数组加锁,将整个数组锁住了,如果多个线程同时访问同一个HashTable对象,就会出现竞争锁,造成锁冲突

缺点: 因为方法加上了锁,所以每次使用的时候都要获取锁,释放锁,性能效率比较低下

🎖️ConcurrentHashMap

底层数据结构为数组+链表+红黑树,红黑树会和链表在某种条件下互相发生转换

⁉️关于ConcurrentHashMap如何实现线程安全:

⏰对于读操作,因为读操作本身就为线程安全的,对于ConcurrentHashMap的属性使用了volatile关键字修饰,确保每次读的值为主存中的最新值

⏲️对于写操作,写操作仍然使用synchronized加锁,但是锁的不是整个数组,如果数组中是链表,锁头结点,如果数组中是红黑树,锁根节点,这样大大降低了锁冲突发生的概率,即多个线程操作不同的结点还是并发并行执行的,操作同一个结点才是互斥的

🕰️充分利用CAS特性,如果结点为空,CAS+自旋,如果结点有元素,synchronized(头节点),size属性也是通过CAS来更新的,这样避免出现重量级锁的情况

🎃对于扩容操作:

⚽需要创建一个新数组,线程发现需要扩容,就搬几个元素到新数组

🏐新老数组是同时存在的

🏀后续的线程也会参与搬几个元素到新数组的过程

🥎待老数组的所有元素都搬完删除老数组

🏉对于扩容的操作,插入只往新数组中插查找需要同时查找新数组和老数组

🎯面试题:HashTable,HashMap,ConcurrentHashMap的区别?

🕐HashMap,线程不安全,key允许为null

🕑HashTable,线程安全,使用synchronized锁整个HashTable对象,效率低,key不允许为null

🕒ConcurrentHashMap,线程安全,对于属性使用了volatile关键字,使用synchronized锁每个链表的头结点降低锁冲突的发生率,充分利用了CAS特性,优化扩容方式,key不允许为null



相关文章
|
3天前
|
消息中间件 监控 Java
滴滴面试:谈谈你对Netty线程模型的理解?
Netty 线程模型是指 Netty 框架为了提供高性能、高并发的网络通信,而设计的管理和利用线程的策略和机制。 **Netty 线程模型被称为 Reactor(响应式)模型/模式,它是基于 NIO 多路复用模型的一种升级,它的核心思想是将 IO 事件和业务处理进行分离,使用一个或多个线程来执行任务的一种机制。** ## 1.**Reactor三大组件** Reactor 包含以下三大组件: ![image.png](https://cdn.nlark.com/yuque/0/2024/png/92791/1717079218890-89000a00-48bc-4a1a-b87e-e1b6
16 2
|
3天前
|
存储 安全 Java
Java中的线程安全集合
【5月更文挑战第30天】在多线程环境下,为了保证数据的一致性和完整性,我们需要使用线程安全的集合。本文将介绍Java中常用的线程安全集合及其使用方法。
|
5天前
|
编解码 安全 算法
Java多线程基础-18:线程安全的集合类与ConcurrentHashMap
如果这些单线程中的集合类确实需要在多线程中使用,该怎么办呢?思路有两个: 最直接的方式:使用锁,手动保证。如多个线程修改ArrayList对象,此时就可能有问题,就可以给修改操作进行加锁。但手动加锁的方式并不是很方便,因此标准库还提供了一些线程安全的集合类。
24 4
|
5天前
|
安全 Java 容器
Java 多线程系列Ⅶ(线程安全集合类)
Java 多线程系列Ⅶ(线程安全集合类)
|
6天前
|
Java 程序员
Java中的多线程编程:理解并应用Thread类和Runnable接口
【5月更文挑战第28天】在Java中,多线程编程是一个重要的概念,它允许同时执行多个任务。本文将深入探讨Java的多线程编程,包括Thread类和Runnable接口的使用,以及如何在实际项目中应用这些知识。我们将通过实例来理解这些概念,并讨论多线程编程的优点和可能的挑战。
|
6天前
|
Java 调度
"ava多线程基础-4:详解Thread类及其基本用法(二)
本文介绍了Java中线程中断和等待的相关概念。
14 0
|
6天前
|
Java Linux API
Java多线程基础-4:详解Thread类及其基本用法 (一)
Java 中的 `Thread` 类是用来管理线程的,每个线程都是通过 `Thread` 类的对象来描述。
26 0
|
6天前
|
设计模式 监控 Java
Java多线程基础-11:工厂模式及代码案例之线程池(一)
本文介绍了Java并发框架中的线程池工具,特别是`java.util.concurrent`包中的`Executors`和`ThreadPoolExecutor`类。线程池通过预先创建并管理一组线程,可以提高多线程任务的效率和响应速度,减少线程创建和销毁的开销。
26 2
|
5天前
|
安全 Linux 编译器
从C语言到C++_40(多线程相关)C++线程接口+线程安全问题加锁(shared_ptr+STL+单例)(下)
从C语言到C++_40(多线程相关)C++线程接口+线程安全问题加锁(shared_ptr+STL+单例)
15 0
|
5天前
|
安全 C语言 C++
从C语言到C++_40(多线程相关)C++线程接口+线程安全问题加锁(shared_ptr+STL+单例)(中)
从C语言到C++_40(多线程相关)C++线程接口+线程安全问题加锁(shared_ptr+STL+单例)
12 0