【Java并发编程】 ConcurrentHashMap 的工作原理?ConcurrentHashMap 和 Hashtable 的区别?

ConcurrentHashMap 的工作原理是什么?

ConcurrentHashMap 是线程安全的哈希表,通过高效的并发控制机制,在并发环境下进行数据存取。ConcurrentHashMap 在 JDK 1.7 和 JDK 1.8 中有两种不同的实现。

  • JDK 1.7 实现:

    • 分段锁(Segment):ConcurrentHashMap 将整个 HashMap 分成多个 Segment,每个 Segment 类似一个小的 HashMap,并且用锁来保护。因此,多个线程可以同时访问不同的 Segment,从而提高并发性能

    • 分段操作:通过对键的哈希值取不同的高位来确定数据存放的 Segment。每个 Segment 独立的加锁操作,减少了锁的粒度,提高了并发性能。

  • JDK 1.8 实现:

    • 使用 CAS 操作和 synchronized 锁来实现操作原子性

    • 哈希桶数组(Node 数组):每个桶存放一个链表(或红黑树,在冲突严重时转换)

    • volatile 关键字:使用 volatile 关键字保证变量的可见性,避免脏数据

    • 简单优先策略:使用 CAS 尝试更新(简单优先),失败时使用 synchronized 锁保护较复杂的操作

基本工作原理:

插入和更新:如果hash计算得到的下标还没有Node,通过 CAS 操作尝试插入或更新。如果有Node,则用synchronized锁住Node,使用 synchronized 锁保护链表或红黑树操作。

扩容:JDK1.7 扩容过程:每个 segment 维护自己的负载因子,当 segment 中的元素数量超过阈值时,该 segment 的 Hashmap会扩容,整体的concurentHashmap 并不是一次性全部扩容。

JDK1.8 基于 CAS 的扩容:在扩容时, concurentHashmap采用了类似 Hashmap 的方式,但通过CAS 操作确保线程安全避免了锁住整个数组。在扩容时,多个线程可以同时帮助完成扩容作。渐进式扩容机制,扩容时并不是一次性将所有数据重新分配,而是多个线程共同参与,逐步迁移旧数据到新数组中,降低了扩容时的性能开销。

ConcurrentHashMap 和 Hashtable 的区别?

ConcurrentHashMap 和 Hashtable 都是线程安全的哈希表实现,但它们在并发控制机制和性能上有显著区别。

  • 锁的粒度:

    • ConcurrentHashMap:采用分段锁(JDK 1.7)或 CAS 和 synchronized(JDK 1.8)。锁的粒度更细,允许更高的并发度,例如 JDK 1.8 中一个桶上锁不会影响其他桶操作

    • Hashtable:对每个方法都加上 synchronized 关键字进行同步,锁的粒度较大。同时只能有一个线程访问整个表,导致高并发性能较差。

  • 性能:

    • ConcurrentHashMap:因为锁的粒度细,并发度高,所以在多线程环境下性能更好。

    • Hashtable:因为锁的粒度大,并发操作频繁时性能较差。

  • 安全性:

    • ConcurrentHashMap:采取了更多措施,例如 CAS 操作、对插入和删除的独立锁定管理,来保证数据的一致性和线程安全。

  • Hashtable:直接通过方法同步来保证,而随着线程增加,锁争用更加激烈,性能下降明显。

  • 迭代器:

    • ConcurrentHashMap:迭代器是弱一致性的,在迭代过程中可修改集合。

    • Hashtable:迭代器在检测到集合被修改时会抛出 ConcurrentModificationException。

**HashMap** * **线程非安全性**:HashMap是非线程安全的,这意味着如果多个线程同时访问并修改HashMap,那么可能会导致数据不一致的问题。它适用于单线程环境或者需要同步控制的操作。 * **键值对存储**:HashMap用于存储键值对(Key-Value pairs),其中每个键都是唯一的,并通过哈希算法快速定位到其对应的值。 * **允许null键和值**:在Java 8及以后版本中,HashMap允许存储null作为键和值。 **Hashtable** * **线程安全性**:Hashtable是线程安全的,意味着它可以同时由多个线程安全地访问而不会引发异常。它适用于多线程环境,无需额外的同步措施。 * **键值对存储**:同样用于存储键值对,但比HashMap更严格,不允许键或值为null。 * **废弃**:从Java 5开始,Hashtable被标记为已过时。现代应用程序通常建议使用Collections.synchronizedMap()函数将其他集合转换为同步的,而非直接使用Hashtable。 **ConcurrentHashMap** * **高并发性能**:ConcurrentHashMap是专门为高并发操作优化的,可以有效地处理大量并发线程的读写操作。 * **分段锁机制**:采用了一种叫做“分段锁”的技术,即将整个映射表分割成多个段,每个段都有自己的锁。这种机制使得多个线程可以在不同的段上并发操作,极大提高了并发性能。 * **允许null键和值**:允许存储null键和值。 * **线程安全性**:虽然ConcurrentHashMap是线程安全的,但它比Hashtable的同步粒度更细,因此在许多情况下提供更好的性能。 **总结**: - **HashMap**:适合单线程或多线程需要外部同步的情况,性能高效,允许null键和值。 - **Hashtable**:已标记为过时,线程安全,不允许null键和值,不推荐新应用中使用。 - **ConcurrentHashMap**:适用于高性能多线程场景,提供了更好的并发性能,允许null键和值。 **相关问题**: 1. HashMapConcurrentHashMap之间的区别是什么? 2. Java中何时应该选择使用 ConcurrentHashMap 而不是 HashMap? 3. 对于单线程应用来说,使用哪种数据结构最合适?
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值