HashMap、ConcurrentHashMap和SynchronizedMap – 哈希表在Java中的多线程同步处理

本文主要是介绍HashMap、ConcurrentHashMap和SynchronizedMap – 哈希表在Java中的多线程同步处理,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

原文链接:http://crunchify.com/hashmap-vs-concurrenthashmap-vs-synchronizedmap-how-a-hashmap-can-be-synchronized-in-java/

在Java中,HashMap是一个非常有用的数据结构。几乎每一个Java应用都会使用到它。我之前的博文中有介绍过如何实现一个线程安全的缓存,在这个例子中,我就使用到了HashMap。然而,需要注意的是,HashMap本身并不是一个线程安全的Collection类

常见问题

  • ConcurrentHashMapCollections.synchronizedMap(Map)分别是什么?
  • ConcurrentHashMapCollections.synchronizedMap(Map)在性能上有什么区别?
  • ConcurrentHashMapCollections.synchronizedMap(Map)的优劣对比?
  • 关于HashMapConcurrentHashMap的常见面试问题

在这篇文章中,将涵盖以上的问题,并解释我们应该如何将HashMap变得线程安全。

原因

Map是一个通过键值对的形势来存储元素的容器,其中,要求key保持唯一,并且每一个key唯一对应一个value。如果你在设计一个高度并行化的程序,并且需要在不同的线程中去读取或者修改一个Map对象,那么使用线程安全的Map则是一个理想的选择。最典型的一个例子就是生产者-消费者模式,生产者不断的修改Map而消费者同时也在读取Map中的值。

那么对于Map来说,什么是线程安全呢?简单的来说,就是当多个线程同时在使用一个Map的时候,至少有一个线程对Map的结构进行了修改,那么必须保证这个修改被立即同步到其他线程中去,避免其他线程获取到错误的值。

解决方案

有两种方法可以解决HashMap的线程安全问题:

  • Java的Collections库中的synchronizedMap()方法
  • 使用ConcurrentHashMap

译者注:其实还有第三种方法,使用Hashtable。不过Hashtable是Java 1.1提供的旧有类,从性能上和使用上都不如其他的替代类,因此已经不推荐使用

//Hashtable
Map<String, String> normalMap = new Hashtable<String, String>();//synchronizedMap
synchronizedHashMap = Collections.synchronizedMap(new HashMap<String, String>());//ConcurrentHashMap
concurrentHashMap = new ConcurrentHashMap<String, String>();

ConcurrentHashMap

  • 当你程序需要高度的并行化的时候,你应该使用ConcurrentHashMap
  • 尽管没有同步整个Map,但是它仍然是线程安全的
  • 读操作非常快,而写操作则是通过加锁完成的
  • 在对象层次上不存在锁(即不会阻塞线程)
  • 锁的粒度设置的非常好,只对哈希表的某一个key加锁
  • ConcurrentHashMap不会抛出ConcurrentModificationException,即使一个线程在遍历的同时,另一个线程尝试进行修改。
  • ConcurrentHashMap会使用多个锁

SynchronizedHashMap

  • 会同步整个对象
  • 每一次的读写操作都需要加锁
  • 对整个对象加锁会极大降低性能
  • 这相当于只允许同一时间内至多一个线程操作整个Map,而其他线程必须等待
  • 它有可能造成资源冲突(某些线程等待较长时间)
  • SynchronizedHashMap会返回Iterator,当遍历时进行修改会抛出异常

示例

  1. 创建类CrunchifyConcurrentHashMapVsSynchronizedHashMap.java
  2. 实例化每一个对象(HashTable, SynchronizedMapConcurrentHashMap
  3. 添加/读取5*50万行数据到Map中
  4. 测试开始和结束的时间
  5. 使用ExecutorService来开启5个线程,同时读写
package crunchify.com.tutorials;import java.util.Collections;
import java.util.HashMap;
import java.util.Hashtable;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.TimeUnit;/*** @author Crunchify.com**/public class CrunchifyConcurrentHashMapVsSynchronizedMap {public final static int THREAD_POOL_SIZE = 5;public static Map<String, Integer> crunchifyHashTableObject = null;public static Map<String, Integer> crunchifySynchronizedMapObject = null;public static Map<String, Integer> crunchifyConcurrentHashMapObject = null;public static void main(String[] args) throws InterruptedException {// Test with Hashtable ObjectcrunchifyHashTableObject = new Hashtable<String, Integer>();crunchifyPerformTest(crunchifyHashTableObject);// Test with synchronizedMap ObjectcrunchifySynchronizedMapObject = Collections.synchronizedMap(new HashMap<String, Integer>());crunchifyPerformTest(crunchifySynchronizedMapObject);// Test with ConcurrentHashMap ObjectcrunchifyConcurrentHashMapObject = new ConcurrentHashMap<String, Integer>();crunchifyPerformTest(crunchifyConcurrentHashMapObject);}public static void crunchifyPerformTest(final Map<String, Integer> crunchifyThreads) throws InterruptedException {System.out.println("Test started for: " + crunchifyThreads.getClass());long averageTime = 0;for (int i = 0; i < 5; i++) {long startTime = System.nanoTime();ExecutorService crunchifyExServer = Executors.newFixedThreadPool(THREAD_POOL_SIZE);for (int j = 0; j < THREAD_POOL_SIZE; j++) {crunchifyExServer.execute(new Runnable() {@SuppressWarnings("unused")@Overridepublic void run() {for (int i = 0; i < 500000; i++) {Integer crunchifyRandomNumber = (int) Math.ceil(Math.random() * 550000);// Retrieve value. We are not using it anywhereInteger crunchifyValue = crunchifyThreads.get(String.valueOf(crunchifyRandomNumber));// Put valuecrunchifyThreads.put(String.valueOf(crunchifyRandomNumber), crunchifyRandomNumber);}}});}// Make sure executor stopscrunchifyExServer.shutdown();// Blocks until all tasks have completed execution after a shutdown requestcrunchifyExServer.awaitTermination(Long.MAX_VALUE, TimeUnit.DAYS);long entTime = System.nanoTime();long totalTime = (entTime - startTime) / 1000000L;averageTime += totalTime;System.out.println("2500K entried added/retrieved in " + totalTime + " ms");}System.out.println("For " + crunchifyThreads.getClass() + " the average time is " + averageTime / 5 + " ms\n");}
}

结果如下:

Test started for: class java.util.Hashtable
500K entried added/retrieved in 1432 ms
500K entried added/retrieved in 1425 ms
500K entried added/retrieved in 1373 ms
500K entried added/retrieved in 1369 ms
500K entried added/retrieved in 1438 ms
For class java.util.Hashtable the average time 1407 msTest started for: class java.util.Collections$SynchronizedMap
500K entried added/retrieved in 1431 ms
500K entried added/retrieved in 1460 ms
500K entried added/retrieved in 1387 ms
500K entried added/retrieved in 1456 ms
500K entried added/retrieved in 1406 ms
For class java.util.Collections$SynchronizedMap the average time 1428 msTest started for: class java.util.concurrent.ConcurrentHashMap
500K entried added/retrieved in 413 ms
500K entried added/retrieved in 351 ms
500K entried added/retrieved in 427 ms
500K entried added/retrieved in 337 ms
500K entried added/retrieved in 339 ms
For class java.util.concurrent.ConcurrentHashMap the average time 373 ms  <== Much faster

    这篇关于HashMap、ConcurrentHashMap和SynchronizedMap – 哈希表在Java中的多线程同步处理的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



    http://www.chinasem.cn/article/898199

    相关文章

    MySQL主从同步延迟问题的全面解决方案

    《MySQL主从同步延迟问题的全面解决方案》MySQL主从同步延迟是分布式数据库系统中的常见问题,会导致从库读取到过期数据,影响业务一致性,下面我将深入分析延迟原因并提供多层次的解决方案,需要的朋友可... 目录一、同步延迟原因深度分析1.1 主从复制原理回顾1.2 延迟产生的关键环节二、实时监控与诊断方案

    SpringBoot中四种AOP实战应用场景及代码实现

    《SpringBoot中四种AOP实战应用场景及代码实现》面向切面编程(AOP)是Spring框架的核心功能之一,它通过预编译和运行期动态代理实现程序功能的统一维护,在SpringBoot应用中,AO... 目录引言场景一:日志记录与性能监控业务需求实现方案使用示例扩展:MDC实现请求跟踪场景二:权限控制与

    Java NoClassDefFoundError运行时错误分析解决

    《JavaNoClassDefFoundError运行时错误分析解决》在Java开发中,NoClassDefFoundError是一种常见的运行时错误,它通常表明Java虚拟机在尝试加载一个类时未能... 目录前言一、问题分析二、报错原因三、解决思路检查类路径配置检查依赖库检查类文件调试类加载器问题四、常见

    Java注解之超越Javadoc的元数据利器详解

    《Java注解之超越Javadoc的元数据利器详解》本文将深入探讨Java注解的定义、类型、内置注解、自定义注解、保留策略、实际应用场景及最佳实践,无论是初学者还是资深开发者,都能通过本文了解如何利用... 目录什么是注解?注解的类型内置注编程解自定义注解注解的保留策略实际用例最佳实践总结在 Java 编程

    Java 实用工具类Spring 的 AnnotationUtils详解

    《Java实用工具类Spring的AnnotationUtils详解》Spring框架提供了一个强大的注解工具类org.springframework.core.annotation.Annot... 目录前言一、AnnotationUtils 的常用方法二、常见应用场景三、与 JDK 原生注解 API 的

    Java controller接口出入参时间序列化转换操作方法(两种)

    《Javacontroller接口出入参时间序列化转换操作方法(两种)》:本文主要介绍Javacontroller接口出入参时间序列化转换操作方法,本文给大家列举两种简单方法,感兴趣的朋友一起看... 目录方式一、使用注解方式二、统一配置场景:在controller编写的接口,在前后端交互过程中一般都会涉及

    Java中的StringBuilder之如何高效构建字符串

    《Java中的StringBuilder之如何高效构建字符串》本文将深入浅出地介绍StringBuilder的使用方法、性能优势以及相关字符串处理技术,结合代码示例帮助读者更好地理解和应用,希望对大家... 目录关键点什么是 StringBuilder?为什么需要 StringBuilder?如何使用 St

    使用Java将各种数据写入Excel表格的操作示例

    《使用Java将各种数据写入Excel表格的操作示例》在数据处理与管理领域,Excel凭借其强大的功能和广泛的应用,成为了数据存储与展示的重要工具,在Java开发过程中,常常需要将不同类型的数据,本文... 目录前言安装免费Java库1. 写入文本、或数值到 Excel单元格2. 写入数组到 Excel表格

    Java并发编程之如何优雅关闭钩子Shutdown Hook

    《Java并发编程之如何优雅关闭钩子ShutdownHook》这篇文章主要为大家详细介绍了Java如何实现优雅关闭钩子ShutdownHook,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起... 目录关闭钩子简介关闭钩子应用场景数据库连接实战演示使用关闭钩子的注意事项开源框架中的关闭钩子机制1.

    Maven中引入 springboot 相关依赖的方式(最新推荐)

    《Maven中引入springboot相关依赖的方式(最新推荐)》:本文主要介绍Maven中引入springboot相关依赖的方式(最新推荐),本文给大家介绍的非常详细,对大家的学习或工作具有... 目录Maven中引入 springboot 相关依赖的方式1. 不使用版本管理(不推荐)2、使用版本管理(推