深度剖析Java HashMap:源码分析、线程安全与最佳实践

简介: 深度剖析Java HashMap:源码分析、线程安全与最佳实践

Java中的HashMap是最常用的数据结构之一,在实际开发中起着至关重要的作用。本文将详细探讨HashMap的工作原理、源码分析、线程安全问题、以及扩容机制等方面。

一、HashMap的基本概念

HashMap是Java集合框架中的一个类,提供了基于哈希表的数据结构。它允许存储键值对,并通过键快速检索对应的值。HashMap允许键和值为null,并且不保证映射的顺序。

二、HashMap的工作原理

HashMap通过哈希函数将键映射到桶(bucket)数组中的一个位置,以实现快速查找。基本操作如put和get的时间复杂度为O(1)。

1. 哈希函数

HashMap使用键的hashCode()方法计算哈希值,然后通过取模运算(hash % array.length)将哈希值映射到数组的索引位置。例如:

int hash = key.hashCode();
int index = (array.length - 1) & hash;
2. 处理哈希冲突

当两个不同的键有相同的哈希值时,会发生哈希冲突。HashMap使用链地址法(separate chaining)处理冲突,即每个桶存储一个链表或红黑树。当链表长度超过阈值(默认为8)时,链表转换为红黑树,以提高查询效率。

三、源码分析

以下是HashMap的核心代码段,包含put方法和get方法。

1. put方法
public V put(K key, V value) {
    return putVal(hash(key), key, value, false, true);
}
 
final V putVal(int hash, K key, V value, boolean onlyIfAbsent, boolean evict) {
    Node<K,V>[] tab; Node<K,V> p; int n, i;
    if ((tab = table) == null || (n = tab.length) == 0)
        n = (tab = resize()).length;
    if ((p = tab[i = (n - 1) & hash]) == null)
        tab[i] = newNode(hash, key, value, null);
    else {
        Node<K,V> e; K k;
        if (p.hash == hash && ((k = p.key) == key || (key != null && key.equals(k))))
            e = p;
        else if (p instanceof TreeNode)
            e = ((TreeNode<K,V>)p).putTreeVal(this, tab, hash, key, value);
        else {
            for (int binCount = 0; ; ++binCount) {
                if ((e = p.next) == null) {
                    p.next = newNode(hash, key, value, null);
                    if (binCount >= TREEIFY_THRESHOLD - 1)
                        treeifyBin(tab, hash);
                    break;
                }
                if (e.hash == hash && ((k = e.key) == key || (key != null && key.equals(k))))
                    break;
                p = e;
            }
        }
        if (e != null) {
            V oldValue = e.value;
            if (!onlyIfAbsent || oldValue == null)
                e.value = value;
            afterNodeAccess(e);
            return oldValue;
        }
    }
    ++modCount;
    if (++size > threshold)
        resize();
    afterNodeInsertion(evict);
    return null;
}
2. get方法
public V get(Object key) {
    Node<K,V> e;
    return (e = getNode(hash(key), key)) == null ? null : e.value;
}
 
final Node<K,V> getNode(int hash, Object key) {
    Node<K,V>[] tab; Node<K,V> first, e; int n; K k;
    if ((tab = table) != null && (n = tab.length) > 0 && (first = tab[(n - 1) & hash]) != null) {
        if (first.hash == hash && ((k = first.key) == key || (key != null && key.equals(k))))
            return first;
        if ((e = first.next) != null) {
            if (first instanceof TreeNode)
                return ((TreeNode<K,V>)first).getTreeNode(hash, key);
            do {
                if (e.hash == hash && ((k = e.key) == key || (key != null && key.equals(k))))
                    return e;
            } while ((e = e.next) != null);
        }
    }
    return null;
}
3. 线程不安全的原因

上述put和get方法在多线程环境中是不安全的。具体原因如下:

put方法线程不安全分析
  1. 扩容(resize):当HashMap需要扩容时,可能会导致多个线程同时进行扩容操作。这会导致数据丢失和不一致。
if (++size > threshold)
    resize();

插入节点(newNode):插入节点时,多个线程可能会同时访问同一个桶位置,导致链表或树结构损坏。

if ((p = tab[i = (n - 1) & hash]) == null)
    tab[i] = newNode(hash, key, value, null);

链表操作:在处理哈希冲突时,链表或红黑树的插入操作不是原子的,可能会导致链表结构损坏。

for (int binCount = 0; ; ++binCount) {
    if ((e = p.next) == null) {
        p.next = newNode(hash, key, value, null);
        if (binCount >= TREEIFY_THRESHOLD - 1)
            treeifyBin(tab, hash);
        break;
    }
    if (e.hash == hash && ((k = e.key) == key || (key != null && key.equals(k))))
        break;
    p = e;
}
get方法线程不安全分析
  1. 读取不一致:在读取节点时,如果另一个线程正在进行插入或删除操作,可能会导致读取的数据不一致。
if ((tab = table) != null && (n = tab.length) > 0 && (first = tab[(n - 1) & hash]) != null) {
    if (first.hash == hash && ((k = first.key) == key || (key != null && key.equals(k))))
        return first;
    if ((e = first.next) != null) {
        if (first instanceof TreeNode)
            return ((TreeNode<K,V>)first).getTreeNode(hash, key);
        do {
            if (e.hash == hash && ((k = e.key) == key || (key != null && key.equals(k))))
                return e;
        } while ((e = e.next) != null);
    }
}

由于这些原因,HashMap在多线程环境中使用时可能会导致不可预测的问题。因此,在多线程环境中,建议使用ConcurrentHashMap替代HashMap

四、线程安全的解决方案
1. 使用ConcurrentHashMap
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
 
public class ConcurrentHashMapExample {
    private static final ConcurrentHashMap<Integer, String> map = new ConcurrentHashMap<>();
 
    public static void main(String[] args) {
        ExecutorService executorService = Executors.newFixedThreadPool(10);
 
        // 使用多个线程并发访问和修改ConcurrentHashMap
        for (int i = 0; i < 100; i++) {
            final int key = i;
            executorService.execute(() -> map.put(key, "Value" + key));
        }
 
        // 读取ConcurrentHashMap中的数据
        executorService.execute(() -> {
            for (int i = 0; i < 100; i++) {
                System.out.println("Key: " + i + ", Value: " + map.get(i));
            }
        });
 
        executorService.shutdown();
    }
}
五、HashMap的初始值设置

在实际开发中,合理设置HashMap的初始容量和负载因子可以提高性能,减少扩容次数。

1. 初始容量

初始容量是HashMap创建时桶数组的大小,默认值为16。初始容量应根据预期的元素数量和负载因子计算:

int initialCapacity = (int) (expectedSize / loadFactor) + 1;

例如,如果预期有100个元素,负载因子为0.75:

int initialCapacity = (int) (100 / 0.75) + 1; // 约等于134
2. 负载因子

负载因子是HashMap在扩容之前允许的最大填充比例,默认值为0.75。负载因子越小,HashMap的空间利用率越低,但查找效率更高。一般情况下,使用默认负载因子即可。

Map<Integer, String> map = new HashMap<>(initialCapacity, 0.75f);

合理设置初始容量和负载因子,可以避免频繁扩容,提高性能。在不确定具体情况时,默认值通常是一个好的选择。

六、HashMap家族中的其他实现

在Java中,除了HashMap,还有其他几个基于哈希表的数据结构实现,它们各自有不同的特点和用途。

1. LinkedHashMap

LinkedHashMap继承自HashMap,并且保留了插入顺序。它使用一个双向链表来维护插入顺序,可以用于需要保持元素顺序的场景。

Map<Integer, String> linkedHashMap = new LinkedHashMap<>();
linkedHashMap.put(1, "one");
linkedHashMap.put(2, "two");
linkedHashMap.put(3, "three");
System.out.println(linkedHashMap); // 输出顺序为1, 2, 3
2. ConcurrentHashMap

ConcurrentHashMap是一个线程安全的HashMap实现,使用了分段锁(segment locking)机制来提高并发性能。适用于高并发场景。

Map<Integer, String> concurrentHashMap = new ConcurrentHashMap<>();
concurrentHashMap.put(1, "one");
concurrentHashMap.put(2, "two");
concurrentHashMap.put(3, "three");
System.out.println(concurrentHashMap);
3. WeakHashMap

WeakHashMap是一种使用弱引用(weak reference)的哈希表实现。其键在没有其他强引用时可以被垃圾回收器回收。适用于缓存和内存敏感的场景。

Map<Integer, String> weakHashMap = new WeakHashMap<>();
Integer key = new Integer(1);
weakHashMap.put(key, "one");
key = null;
System.gc();
System.out.println(weakHashMap); // 可能为空,因为key可能被回收
4. IdentityHashMap

IdentityHashMap使用键的引用相等性(reference equality)而不是键的equals方法来比较键。适用于需要比较对象引用而不是对象内容的场景。

Map<Integer, String> identityHashMap = new IdentityHashMap<>();
identityHashMap.put(new Integer(1), "one");
identityHashMap.put(new Integer(1), "one again");
System.out.println(identityHashMap.size()); // 输出2

目录
打赏
0
1
1
0
28
分享
相关文章
|
5天前
|
【源码】【Java并发】【线程池】邀请您从0-1阅读ThreadPoolExecutor源码
当我们创建一个`ThreadPoolExecutor`的时候,你是否会好奇🤔,它到底发生了什么?比如:我传的拒绝策略、线程工厂是啥时候被使用的? 核心线程数是个啥?最大线程数和它又有什么关系?线程池,它是怎么调度,我们传入的线程?...不要着急,小手手点上关注、点赞、收藏。主播马上从源码的角度带你们探索神秘线程池的世界...
50 0
【源码】【Java并发】【线程池】邀请您从0-1阅读ThreadPoolExecutor源码
|
9天前
|
【Java并发】【线程池】带你从0-1入门线程池
欢迎来到我的技术博客!我是一名热爱编程的开发者,梦想是编写高端CRUD应用。2025年我正在沉淀中,博客更新速度加快,期待与你一起成长。 线程池是一种复用线程资源的机制,通过预先创建一定数量的线程并管理其生命周期,避免频繁创建/销毁线程带来的性能开销。它解决了线程创建成本高、资源耗尽风险、响应速度慢和任务执行缺乏管理等问题。
119 60
【Java并发】【线程池】带你从0-1入门线程池
Java HashMap详解及实现原理
Java HashMap是Java集合框架中常用的Map接口实现,基于哈希表结构,允许null键和值,提供高效的存取操作。它通过哈希函数将键映射到数组索引,并使用链表或红黑树解决哈希冲突。HashMap非线程安全,多线程环境下需注意并发问题,常用解决方案包括ConcurrentHashMap和Collections.synchronizedMap()。此外,合理设置初始化容量和加载因子、重写hashCode()和equals()方法有助于提高性能和避免哈希冲突。
49 17
Java HashMap详解及实现原理
Java社招面试题:一个线程运行时发生异常会怎样?
大家好,我是小米。今天分享一个经典的 Java 面试题:线程运行时发生异常,程序会怎样处理?此问题考察 Java 线程和异常处理机制的理解。线程发生异常,默认会导致线程终止,但可以通过 try-catch 捕获并处理,避免影响其他线程。未捕获的异常可通过 Thread.UncaughtExceptionHandler 处理。线程池中的异常会被自动处理,不影响任务执行。希望这篇文章能帮助你深入理解 Java 线程异常处理机制,为面试做好准备。如果你觉得有帮助,欢迎收藏、转发!
86 14
Java 面试必问!线程构造方法和静态块的执行线程到底是谁?
大家好,我是小米。今天聊聊Java多线程面试题:线程类的构造方法和静态块是由哪个线程调用的?构造方法由创建线程实例的主线程调用,静态块在类加载时由主线程调用。理解这些细节有助于掌握Java多线程机制。下期再见! 简介: 本文通过一个常见的Java多线程面试题,详细讲解了线程类的构造方法和静态块是由哪个线程调用的。构造方法由创建线程实例的主线程调用,静态块在类加载时由主线程调用。理解这些细节对掌握Java多线程编程至关重要。
53 13
【JAVA】封装多线程原理
Java 中的多线程封装旨在简化使用、提高安全性和增强可维护性。通过抽象和隐藏底层细节,提供简洁接口。常见封装方式包括基于 Runnable 和 Callable 接口的任务封装,以及线程池的封装。Runnable 适用于无返回值任务,Callable 支持有返回值任务。线程池(如 ExecutorService)则用于管理和复用线程,减少性能开销。示例代码展示了如何实现这些封装,使多线程编程更加高效和安全。
|
24天前
|
Linux编程: 在业务线程中注册和处理Linux信号
本文详细介绍了如何在Linux中通过在业务线程中注册和处理信号。我们讨论了信号的基本概念,并通过完整的代码示例展示了在业务线程中注册和处理信号的方法。通过正确地使用信号处理机制,可以提高程序的健壮性和响应能力。希望本文能帮助您更好地理解和应用Linux信号处理,提高开发效率和代码质量。
45 17
|
1月前
|
Linux编程: 在业务线程中注册和处理Linux信号
通过本文,您可以了解如何在业务线程中注册和处理Linux信号。正确处理信号可以提高程序的健壮性和稳定性。希望这些内容能帮助您更好地理解和应用Linux信号处理机制。
55 26
【JavaEE】多线程编程引入——认识Thread类
Thread类,Thread中的run方法,在编程中怎么调度多线程
|
3月前
|
Java多线程编程秘籍:各种方案一网打尽,不要错过!
Java 中实现多线程的方式主要有四种:继承 Thread 类、实现 Runnable 接口、实现 Callable 接口和使用线程池。每种方式各有优缺点,适用于不同的场景。继承 Thread 类最简单,实现 Runnable 接口更灵活,Callable 接口支持返回结果,线程池则便于管理和复用线程。实际应用中可根据需求选择合适的方式。此外,还介绍了多线程相关的常见面试问题及答案,涵盖线程概念、线程安全、线程池等知识点。
269 2

热门文章

最新文章

AI助理

你好,我是AI助理

可以解答问题、推荐解决方案等