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
ConcurrentHashMap底层实现原理
ConcurrentHashMap底层实现原理
295 0
|
存储 算法 安全
HashMap底层实现原理
HashMap底层实现原理
170 0
|
2月前
|
存储 Java 索引
HashMap原理详解,包括底层原理
【11月更文挑战第14天】本文介绍了数据结构基础,重点讲解了哈希表的概念及其实现方式,包括数组与链表的特点及其在HashMap中的应用。详细分析了Java 7及Java 8版本中HashMap的底层存储结构,特别是Java 8中引入的红黑树优化。此外,还探讨了哈希函数的设计、哈希冲突的解决策略以及HashMap的重要方法实现原理,如put、get和remove方法,最后讨论了HashMap的容量与扩容机制。
|
5月前
|
Java 调度
【多线程面试题十四】、说一说synchronized的底层实现原理
这篇文章解释了Java中的`synchronized`关键字的底层实现原理,包括它在代码块和方法同步中的实现方式,以及通过`monitorenter`和`monitorexit`指令以及`ACC_SYNCHRONIZED`访问标志来控制线程同步和锁的获取与释放。
|
8月前
|
存储 安全 Java
CopyOnWriteArrayList底层原理全面解析【建议收藏】
CopyOnWriteArrayList是Java中的一个线程安全的集合类,是ArrayList线程安全版本,主要通过Copy-On-Write(写时复制,简称COW)机制来保证线程安全。 Copy-On-Write机制核心思想:向一个数组中添加数据时,不直接操作原始数组,而是拷贝原始数组生成一份原始数组副本,将需要添加的数据添加到原始数组副本中,操作完成后再用原始数组副本直接替换原始数组,从而保证多个线程同时操作原始数组时的线程安全。
|
安全 Java
ArrayList底层实现原理
ArrayList底层实现原理
79 0
|
存储 算法 Java
从HashMap的执行流程开始 揭开HashMap底层实现
从HashMap的执行流程开始 揭开HashMap底层实现
42 0
|
存储 Java
Synchronized 用法和底层原理
Synchronized 用法和底层原理
175 1
|
存储 算法 安全
【多线程系列-05】深入理解ThreadLocal的底层原理和基本使用
【多线程系列-05】深入理解ThreadLocal的底层原理和基本使用
272 4
|
安全
多线程环境下,如何实现一个 Hash 结构?介绍一下 ConCurrentHashMap 的底层原理?
多线程环境下,如何实现一个 Hash 结构?介绍一下 ConCurrentHashMap 的底层原理?
84 0
多线程环境下,如何实现一个 Hash 结构?介绍一下 ConCurrentHashMap 的底层原理?