HashMap在多线程环境下不安全,主要表现为JDK 1.7中put扩容引发的死循环、各版本均存在的数据覆盖与丢失、结构性修改导致的竞态条件及迭代异常;推荐使用ConcurrentHashMap替代。
HashMap在多线程环境下是不安全的,这主要体现在多个线程同时操作同一个HashMap实例时可能出现数据错乱、死循环、甚至程序崩溃等问题。下面从几个关键方面分析其不安全的原因。
在JDK 1.7中,HashMap使用头插法进行链表插入。当多个线程同时触发扩容(resize)时,可能会形成环形链表,从而在get操作时造成死循环。
问题场景:
虽然JDK 1.8改用尾插法解决了这个问题,但并不意味着HashMap变得线程安全了。
多个线程同时执行put操作时,可能因为缺乏同步机制导致数据被覆盖。
典型情况:
这是因为put操作不是原子的:包括查找、创建节点、链接等多个步骤,中间状态可能被其他线程干扰。
结构性修改如put、remove、clear等,在并发下可能导致内部结构不一致。
例如:
虽然fail-fast能及时发现问题,但它只是“检测”错误,并不能防止错误发生。
为解决HashMap的线程安全问题,Java提供了以下替代选择:
基本上就这些。HashMap设计初衷就是非线程安全的,追求高性能。多线程环境下必须使用 ConcurrentHashMap 或采取外部同步措施,否则极易引发难以排查的问题。不复杂但容易忽略。