ConcurrentHashMap:使用方法和底层原理详解

简介: ConcurrentHashMap:使用方法和底层原理详解

ConcurrentHashMap:使用方法和底层原理详解


在Java编程中,ConcurrentHashMap是一个非常强大和常用的数据结构,用于在多线程环境下安全地操作Map。本文将深入探讨ConcurrentHashMap的各种使用方法以及其底层原理。


1. ConcurrentHashMap简介


ConcurrentHashMap是Java集合框架中的一个线程安全的哈希表实现,它提供了比Hashtable和同步的HashMap更高的并发性能。ConcurrentHashMap在JDK 1.5中引入,通过采用分段锁的方式实现了高效的并发访问。它可以在保证线程安全的同时,提供较高的并发性能,因此是多线程环境下使用频率较高的数据结构之一。


2. ConcurrentHashMap的使用方法


2.1 添加和获取元素


ConcurrentHashMap的使用方式与HashMap类似,可以通过put(key, value)方法添加元素,通过get(key)方法获取元素。

ConcurrentHashMap<String, Integer> map = new ConcurrentHashMap<>();
map.put("key1", 1);
map.put("key2", 2);
int value = map.get("key1");

2.2 删除元素


使用remove(key)方法可以从ConcurrentHashMap中删除指定的键值对。

map.remove("key1");


2.3 遍历元素


ConcurrentHashMap提供了多种遍历方式,如迭代器遍历、forEach遍历等。

// 使用迭代器遍历
Iterator<Map.Entry<String, Integer>> iterator = map.entrySet().iterator();
while (iterator.hasNext()) {
    Map.Entry<String, Integer> entry = iterator.next();
    String key = entry.getKey();
    Integer value = entry.getValue();
    // 处理每个键值对
}

// 使用forEach遍历
map.forEach((key, value) -> {
    // 处理每个键值对
});

这些方法是ConcurrentHashMap提供的一些其他常用方法,用于在特定条件下添加、更新或计算元素。让我们逐一来讲解它们的作用和使用方法:


2.4putIfAbsent(key, value)


putIfAbsent(key, value)方法用于将指定的键值对添加到ConcurrentHashMap中,但只有在指定的键尚未存在时才添加。如果已存在相同的键,则不进行添加操作,并返回该键对应的原始值。

import java.util.concurrent.ConcurrentHashMap;

public class ConcurrentHashMapExample {
    public static void main(String[] args) {
        ConcurrentHashMap<String, Integer> map = new ConcurrentHashMap<>();
        map.put("key1", 1);

        Integer oldValue = map.putIfAbsent("key1", 2);
        System.out.println("Old value for key1: " + oldValue); // 1
    }
}

在上述示例中,由于已存在key为"key1"的键值对,所以putIfAbsent方法不会添加新的键值对,而是返回原始值1。


2.5 computeIfAbsent(key, mappingFunction)


computeIfAbsent(key, mappingFunction)方法用于根据指定的键计算一个值,并将其与指定的键关联。如果指定的键已存在(或映射到null),则不执行映射函数。

import java.util.concurrent.ConcurrentHashMap;

public class ConcurrentHashMapExample {
    public static void main(String[] args) {
        ConcurrentHashMap<String, Integer> map = new ConcurrentHashMap<>();
        map.put("key1", 1);

        map.computeIfAbsent("key2", key -> key.length());
        System.out.println("Value for key2: " + map.get("key2")); // 4
    }
}

在上述示例中,由于"key2"键尚不存在于ConcurrentHashMap中,因此会根据映射函数key -> key.length()计算出值4,并将"key2"与其关联。


2.6 computeIfPresent(key, remappingFunction)


computeIfPresent(key, remappingFunction)方法用于根据指定的键及其当前映射值计算一个新的映射值,并将其与指定的键关联。如果指定的键不存在或其值为null,则不执行映射函数。

import java.util.concurrent.ConcurrentHashMap;

public class ConcurrentHashMapExample {
    public static void main(String[] args) {
        ConcurrentHashMap<String, Integer> map = new ConcurrentHashMap<>();
        map.put("key1", 1);

        map.computeIfPresent("key1", (key, value) -> value * 10);
        System.out.println("New value for key1: " + map.get("key1")); // 10
    }
}

在上述示例中,由于"key1"键存在于ConcurrentHashMap中,因此会根据映射函数(key, value) -> value * 10计算出新的值10,并将其与"key1"关联。


这些方法的使用方式类似于HashMap中的相应方法,但在多线程环境中更加安全。通过合理地使用这些方法,我们可以更加灵活地操作ConcurrentHashMap,并在多线程环境下保证数据的一致性和线程安全性。


ConcurrentHashMap的底层原理详解


1. 分段锁机制


ConcurrentHashMap将整个哈希表分成多个段(Segment),每个段都是一个独立的哈希表,相互之间不会产生影响。这些段之间是相互独立的,每个段都拥有自己的锁。当执行put或get操作时,首先根据key的哈希值确定它所属的段,然后在该段上加锁,其他线程可以并发地访问其他段。这样一来,不同的线程可以同时操作不同的段,大大提高了并发性能。


2. 案例分析


假设我们有一个需要高并发访问的缓存系统,使用ConcurrentHashMap来存储缓存数据。在这个缓存系统中,我们希望能够并发地添加、获取和删除缓存数据,而不会出现线程安全问题。

import java.util.concurrent.ConcurrentHashMap;

public class CacheSystem {
    private ConcurrentHashMap<String, String> cacheMap = new ConcurrentHashMap<>();

    public void put(String key, String value) {
        cacheMap.put(key, value);
    }

    public String get(String key) {
        return cacheMap.get(key);
    }

    public void remove(String key) {
        cacheMap.remove(key);
    }
}

在上述代码中,我们创建了一个名为CacheSystem的类,其中的cacheMap是一个ConcurrentHashMap实例,用于存储缓存数据。put、get和remove方法分别用于添加、获取和删除缓存数据。


3. 分析底层原理


当多个线程同时访问缓存系统时,ConcurrentHashMap会根据key的哈希值将其分配到不同的段上,并在该段上加锁。这样即使多个线程同时操作不同的key,它们也可以并发地访问不同的段,不会产生线程安全问题。


例如,当线程A调用put方法添加一个缓存数据时,ConcurrentHashMap会根据key的哈希值确定该数据所属的段,并在该段上加锁,确保其他线程无法同时修改该段的数据。而同时,线程B可能在另一个段上执行get方法获取缓存数据,由于各个段之间是相互独立的,因此不会受到线程A的影响,可以并发地执行get操作。


这种分段锁机制有效地降低了并发操作的锁竞争,提高了系统的并发性能。


相关文章
|
Java Spring 容器
Spring的AOP失效场景详解
Spring的AOP失效场景详解
1752 0
|
存储 缓存 安全
ConcurrentHashMap的实现原理,非常详细,一文吃透!
本文详细解析了ConcurrentHashMap的实现原理,深入探讨了分段锁、CAS操作和红黑树等关键技术,帮助全面理解ConcurrentHashMap的并发机制。关注【mikechen的互联网架构】,10年+BAT架构经验倾囊相授。
ConcurrentHashMap的实现原理,非常详细,一文吃透!
太惨痛: Redis 分布式锁 5个大坑,又大又深, 如何才能 避开 ?
Redis分布式锁在高并发场景下是重要的技术手段,但其实现过程中常遇到五大深坑:**原子性问题**、**连接耗尽问题**、**锁过期问题**、**锁失效问题**以及**锁分段问题**。这些问题不仅影响系统的稳定性和性能,还可能导致数据不一致。尼恩在实际项目中总结了这些坑,并提供了详细的解决方案,包括使用Lua脚本保证原子性、设置合理的锁过期时间和使用看门狗机制、以及通过锁分段提升性能。这些经验和技巧对面试和实际开发都有很大帮助,值得深入学习和实践。
太惨痛: Redis 分布式锁 5个大坑,又大又深, 如何才能 避开 ?
|
缓存 安全 Java
全面解读ConcurrentHashMap:Java中的高效并发数据结构
全面解读ConcurrentHashMap:Java中的高效并发数据结构
2501 2
|
Kubernetes 测试技术 数据库
详解微服务应用灰度发布最佳实践
相对于传统软件研发,微服务架构下典型的需求交付最大的区别在于有了能够小范围真实验证的机制,且交付单位较小,风险可控,灰度发布可以弥补线下测试的不足。本文从 DevOps 视角概述灰度发布实践,介绍如何将灰度发布与 DevOps 工作融合,快来了解吧~
33302 19
|
负载均衡 监控 Dubbo
Dubbo 原理和机制详解(非常全面)
本文详细解析了 Dubbo 的核心功能、组件、架构设计及调用流程,涵盖远程方法调用、智能容错、负载均衡、服务注册与发现等内容。欢迎留言交流。关注【mikechen的互联网架构】,10年+BAT架构经验倾囊相授。
Dubbo 原理和机制详解(非常全面)
|
消息中间件 存储 Java
吃透 RocketMQ 消息中间件,看这篇就够了!
本文详细介绍 RocketMQ 的五大要点、核心特性及应用场景,涵盖高并发业务场景下的消息中间件关键知识点。关注【mikechen的互联网架构】,10年+BAT架构经验倾囊相授。
吃透 RocketMQ 消息中间件,看这篇就够了!
|
SQL XML JavaScript
【若依Java】15分钟玩转若依二次开发,新手小白半小时实现前后端分离项目,springboot+vue3+Element Plus+vite实现Java项目和管理后台网站功能
摘要: 本文档详细介绍了如何使用若依框架快速搭建一个基于SpringBoot和Vue3的前后端分离的Java管理后台。教程涵盖了技术点、准备工作、启动项目、自动生成代码、数据库配置、菜单管理、代码下载和导入、自定义主题样式、代码生成、启动Vue3项目、修改代码、以及对代码进行自定义和扩展,例如单表和主子表的代码生成、树形表的实现、商品列表和分类列表的改造等。整个过程详细地指导了如何从下载项目到配置数据库,再到生成Java和Vue3代码,最后实现前后端的运行和功能定制。此外,还提供了关于软件安装、环境变量配置和代码自动生成的注意事项。
28441 73
mybatis复习01,简单配置让mybatis跑起来
文章介绍了MyBatis的基本概念、历史和特点,并详细指导了如何配置MyBatis环境,包括创建Maven项目、添加依赖、编写核心配置文件、创建数据表和实体类、编写Mapper接口和XML配置文件,以及如何编写工具类和测试用例。
mybatis复习01,简单配置让mybatis跑起来

热门文章

最新文章