【亮剑】Java中的并发容器ConcurrentHashMap,它在JDK1.5中引入,用于替换HashTable和SynchronizedMap

简介: 【4月更文挑战第30天】本文介绍了Java中的并发容器ConcurrentHashMap,它在JDK1.5中引入,用于替换HashTable和SynchronizedMap。文章展示了创建、添加、获取、删除和遍历元素的基本用法。ConcurrentHashMap的内部实现基于分段锁,每个段是一个独立的Hash表,通过分段锁实现并发控制。每个段内部采用数组+链表/红黑树的数据结构,当冲突过多时转为红黑树优化查询。此外,它有扩容机制,当元素超过阈值时,会逐段扩容并翻倍Segment数量,以保持高性能的并发访问。

一、引言

在多线程环境下,为了保证数据的一致性和并发性能,我们通常会使用到并发容器。而在Java中,ConcurrentHashMap就是一种非常常用的并发容器。它是JDK1.5中引入的,用于替代HashTable和SynchronizedMap等同步容器,提供了更好的并发性能。本文将详细介绍ConcurrentHashMap的使用方法及其内部实现原理。

二、ConcurrentHashMap的使用方法

  1. 创建ConcurrentHashMap对象
import java.util.concurrent.ConcurrentHashMap;

public class ConcurrentHashMapDemo {
   
    public static void main(String[] args) {
   
        ConcurrentHashMap<String, Integer> concurrentHashMap = new ConcurrentHashMap<>();
    }
}
  1. 添加元素
import java.util.concurrent.ConcurrentHashMap;

public class ConcurrentHashMapDemo {
   
    public static void main(String[] args) {
   
        ConcurrentHashMap<String, Integer> concurrentHashMap = new ConcurrentHashMap<>();
        concurrentHashMap.put("one", 1);
        concurrentHashMap.put("two", 2);
        concurrentHashMap.put("three", 3);
    }
}
  1. 获取元素
import java.util.concurrent.ConcurrentHashMap;

public class ConcurrentHashMapDemo {
   
    public static void main(String[] args) {
   
        ConcurrentHashMap<String, Integer> concurrentHashMap = new ConcurrentHashMap<>();
        concurrentHashMap.put("one", 1);
        concurrentHashMap.put("two", 2);
        concurrentHashMap.put("three", 3);

        Integer one = concurrentHashMap.get("one");
        System.out.println("one: " + one);
    }
}
  1. 删除元素
import java.util.concurrent.ConcurrentHashMap;

public class ConcurrentHashMapDemo {
   
    public static void main(String[] args) {
   
        ConcurrentHashMap<String, Integer> concurrentHashMap = new ConcurrentHashMap<>();
        concurrentHashMap.put("one", 1);
        concurrentHashMap.put("two", 2);
        concurrentHashMap.put("three", 3);

        concurrentHashMap.remove("one");
    }
}
  1. 遍历元素
import java.util.concurrent.ConcurrentHashMap;

public class ConcurrentHashMapDemo {
   
    public static void main(String[] args) {
   
        ConcurrentHashMap<String, Integer> concurrentHashMap = new ConcurrentHashMap<>();
        concurrentHashMap.put("one", 1);
        concurrentHashMap.put("two", 2);
        concurrentHashMap.put("three", 3);

        for (String key : concurrentHashMap.keySet()) {
   
            System.out.println("Key: " + key + ", Value: " + concurrentHashMap.get(key));
        }
    }
}

三、ConcurrentHashMap的内部实现原理

  1. 分段锁(Segment)

ConcurrentHashMap的核心思想是将整个Map分为N个Segment,每个Segment独立维护一部分数据。这样在进行put、remove等操作时,只需要锁定当前Segment,而不需要锁定整个Map,从而提高并发性能。默认情况下,ConcurrentHashMap会创建16个Segment。

  1. 数据结构

每个Segment内部使用了一个数组来存储键值对,数组的索引通过hashCode计算得到。当发生哈希冲突时,会使用链表或红黑树来解决。当链表长度超过一定阈值时,会转换为红黑树以提高查询效率。

  1. 扩容机制

当某个Segment的元素个数超过阈值时,会触发扩容操作。扩容时,会创建一个新的数组,并将旧数组的数据迁移到新数组中。同时,Segment的数量也会翻倍。为了减少锁的粒度,ConcurrentHashMap采用了逐段扩容的策略,每次只扩容一个Segment。

四、总结

本文详细介绍了ConcurrentHashMap的使用方法及其内部实现原理。通过分段锁的设计,ConcurrentHashMap在保证数据一致性的同时,提供了较好的并发性能。在实际应用中,我们可以根据需要选择合适的并发容器,以提高系统的性能。

相关文章
|
1月前
|
Java 虚拟化 容器
(Java)Java里JFrame窗体的基本操作(容器布局篇-1)
容器 容器,我的理解是可以包容其他东西的玩意。它可以是一个盒子,可以是一个虚拟化的物品,可只要能包裹住其他存在质体的东西,那么都可以称作是容器。例如:JPanel组件和JScollPane组件两者都是容器也是组件。 既然有容器,那么容器中的布局就必不可少了。不然不规矩的摆放物品,人类看不习惯,我也看不习惯 ???? 本篇内容,将说明java JFrame窗体里容器中几类布局。 说明:所有在JFrame窗体里的容器布局都会使用setLayout()方法,采用的布局参数都将放进这个方法里 绝对布局 调用窗体容器
63 1
|
24天前
|
Java 大数据 Go
从混沌到秩序:Java共享内存模型如何通过显式约束驯服并发?
并发编程旨在混乱中建立秩序。本文对比Java共享内存模型与Golang消息传递模型,剖析显式同步与隐式因果的哲学差异,揭示happens-before等机制如何保障内存可见性与数据一致性,展现两大范式的深层分野。(238字)
44 4
|
26天前
|
缓存 安全 Java
如何理解Java中的并发?
Java并发指多任务交替执行,提升资源利用率与响应速度。通过线程实现,涉及线程安全、可见性、原子性等问题,需用synchronized、volatile、线程池及并发工具类解决,是高并发系统开发的关键基础。(238字)
132 4
|
3月前
|
安全 Oracle Java
JAVA高级开发必备·卓伊凡详细JDK、JRE、JVM与Java生态深度解析-形象比喻系统理解-优雅草卓伊凡
JAVA高级开发必备·卓伊凡详细JDK、JRE、JVM与Java生态深度解析-形象比喻系统理解-优雅草卓伊凡
270 0
JAVA高级开发必备·卓伊凡详细JDK、JRE、JVM与Java生态深度解析-形象比喻系统理解-优雅草卓伊凡
|
4月前
|
SQL 缓存 安全
深度理解 Java 内存模型:从并发基石到实践应用
本文深入解析 Java 内存模型(JMM),涵盖其在并发编程中的核心作用与实践应用。内容包括 JMM 解决的可见性、原子性和有序性问题,线程与内存的交互机制,volatile、synchronized 和 happens-before 等关键机制的使用,以及在单例模式、线程通信等场景中的实战案例。同时,还介绍了常见并发 Bug 的排查与解决方案,帮助开发者写出高效、线程安全的 Java 程序。
221 0
|
4月前
|
安全 Java 微服务
Java 最新技术和框架实操:涵盖 JDK 21 新特性与 Spring Security 6.x 安全框架搭建
本文系统整理了Java最新技术与主流框架实操内容,涵盖Java 17+新特性(如模式匹配、文本块、记录类)、Spring Boot 3微服务开发、响应式编程(WebFlux)、容器化部署(Docker+K8s)、测试与CI/CD实践,附完整代码示例和学习资源推荐,助你构建现代Java全栈开发能力。
475 0
|
4月前
|
Oracle Java 关系型数据库
新手必看:Java 开发环境搭建之 JDK 与 Maven
本文分享了 Java 学习中 JDK 安装配置与 Maven 使用的入门知识,涵盖 JDK 下载安装、环境变量设置、Maven 安装配置及本地仓库与镜像设置,帮助新手快速搭建 Java 开发环境。
398 0
|
安全 算法 Java
【Java集合类面试二】、 Java中的容器,线程安全和线程不安全的分别有哪些?
这篇文章讨论了Java集合类的线程安全性,列举了线程不安全的集合类(如HashSet、ArrayList、HashMap)和线程安全的集合类(如Vector、Hashtable),同时介绍了Java 5之后提供的java.util.concurrent包中的高效并发集合类,如ConcurrentHashMap和CopyOnWriteArrayList。
【Java集合类面试二】、 Java中的容器,线程安全和线程不安全的分别有哪些?
|
存储 Java 容器
Java中集合容器详解:简单使用与案例分析3
Java中集合容器详解:简单使用与案例分析
192 0
|
安全 算法 Java
安全无忧:Java并发集合容器的应用与实践
安全无忧:Java并发集合容器的应用与实践
124 0
安全无忧:Java并发集合容器的应用与实践