Java并发容器篇
作者:汤圆
个人博客:javalover.cc
前言
断断续续一个多月,也写了十几篇原创文章,感觉真的很不一样;
不能说技术有很大的进步,但是想法确实跟以前有所不同;
还没开始的时候,想着要学的东西太多,总觉得无从下手;
但是当你真正下定决心去做了几天后,就会发现 原来路真的是一步步走出来的;
如果总是原地踏步东张西望,对自己不会有帮助;
好了,下面开始今天的话题,并发容器篇
简介
前面我们介绍了同步容器,它的很大一个缺点就是在高并发下的环境下,性能差;
针对这个,于是就有了专门为高并发设计的并发容器类;
因为并发容器类都位于java.util.concurrent下,所以我们也习惯把并发容器简称为JUC容器;
相对应的还有JUC原子类、JUC锁、JUC工具类等等(这些后面再介绍)
今天就让我们简单来了解下JUC中并发容器的相关知识点
文章如果有问题,欢迎大家批评指正,在此谢过啦
目录
- 什么是并发容器
- 为什么会有并发容器
- 并发容器、同步容器、普通容器的区别
正文
1. 什么是并发容器
并发容器是针对高并发专门设计的一些类,用来替代性能较低的同步容器
常见的并发容器类如下所示:

这节我们主要以第一个ConcurrentHashMap为例子来介绍并发容器
其他的以后有空会单独开篇分析
2. 为什么会有并发容器
其实跟同步容器的出现的道理是一样的:
同步容器是为了让我们在编写多线程代码时,不用自己手动去同步加锁,为我们解放了双手,去做更多有意义的事情(有意义?双手?);
而并发容器则又是为了提高同步容器的性能,相当于同步容器的升级版;
这也是为什么Java一直在被人唱衰,却又一直没有衰退的原因(大佬们也很焦虑啊!!!);
不过话说回来,大佬们焦虑地有点过头了;不敢想Java现在都升到16级了,而我们始终还在8级徘徊。
3. 并发容器、同步容器、普通容器的区别
这里的普通容器,指的是没有同步和并发的容器类,比如HashMap
三个对比着来介绍,这样会更加清晰一点
下面我们分别以HashMap, HashTable, ConcurrentHashMap为例来介绍
性能分析
下面我们来分析下他们三个之间的性能区别:
注:这里普通容器用的是单线程来测试的,因为多线程不安全,所以我们就不考虑了
有的朋友可能会说,你这不公平啊,可是没办法呀,谁让她多线程不安全呢。
如果非要让我在安全和性能之间选一个的话,那我选 ConcurrentHashMap(我都要)
他们三个之间的关系,如下图

(红色表示堵的厉害,橙色表示堵的一般,绿色表示畅通)
可以看到:
在单线程中操作普通容器时,代码都是串行执行的,同一时刻只能put或get一个数据到容器中
在多线程中操作同步容器时,可以多个线程排队去执行,同一时刻也是只能put或get一个数据到同步容器中
在多线程中操作并发容器时,可以多个线程同时去执行,也就是说同一时刻可以有多个线程去put或get多个数据到并发容器中(可同时读读,可同时读写,可同时写写-有可能会阻塞,这里是以ConcurrentHashMap为参考)
下面我们用代码来复现下上面图中所示的效果(慢-中-快)
- HashMap 测试方法
public static void hashMapTest(){
Map<String, String> map = new HashMap<>();
long start = System.nanoTime();
// 创建10万条数据 单线程
for (int i = 0; i < 100_000; i++) {
// 用UUID作为key,保证key的唯一
map.put(UUID.randomUUID().toString(), String.valueOf(i));
map.get(UUID.randomUUID().toString());
}
long end = System.nanoTime();
System.out.println("hashMap耗时:");
System.out.println(end - start);
}
- HashTable 测试方法
public static void hashTableTest(){
Map<String, String> map = new Hashtable<>();
long start = System.nanoTime();
// 创建10个线程 - 多线程
for (int i = 0; i < 10; i++) {
new Thread(()->{
// 每个线程创建1万条数据
for (int j = 0; j < 10000; j++) {
// UUID保证key的唯一性
map.put(UUID.randomUUID().toString(), String.valueOf(j));
map.get(UUID.randomUUID().toString());
}
}).start();
}
// 这里是为了等待上面的线程执行结束,之所以判断>2,是因为在IDEA中除了main thread,还有一个monitor thread
while (Thread.activeCount()>2){
Thread.yield();
}
long end = System.nanoTime();
System.out.println("hashTable耗时:");
System.out.println(end - start);
}
- concurrentHashMap 测试方法
public static void concurrentHashMapTest(){
Map<String, String> map = new ConcurrentHashMap<>();
long start = System.nanoTime();
// 创建10个线程 - 多线程
for (int i = 0; i < 10; i++) {
new Thread(()->{
// 每个线程创建1万条数据
for (int j = 0; j < 10000; j++) {
// UUID作为key,保证唯一性
map.put(UUID.randomUUID().toString(), String.valueOf(j));
map.get(UUID.randomUUID().toString());
}
}).start();
}
// 这里是为了等待上面的线程执行结束,之所以判断>2,是因为在IDEA中除了main thread,还有一个monitor thread
while (Thread.activeCount()>2){
Thread.yield();
}
long end = System.nanoTime();
System.out.println("concurrentHashMap耗时:");
System.out.println(end - start);
}
- main 方法分别执行上面的三个测试
public static void main(String[] args) {
hashMapTest();
hashTableTest();
while (Thread.activeCount()>2){
Thread.yield();
}
concurrentHashMapTest();
}
运行可以看到,如下结果(运行多次,数值可能会变好,但是规律基本一致)
hashMap耗时:
754699874 (慢)
hashTable耗时:
609160132(中)
concurrentHashMap耗时:
261617133(快)
结论就是,正常情况下的速度:普通容器 < 同步容器 < 并发容器
但是也不那么绝对,因为这里插入的key都是唯一的,所以看起来正常一点
那如果我们不正常一点呢?比如极端到BT的那种
下面我们就不停地插入同一条数据,上面的所有put/get都改为下面的代码:
map.put("a", "a");
map.get("a");
运行后,你会发现,又是另外一个结论(大家感兴趣的可以敲出来试试)
不过结论不结论的,意义不是很大;
锁分析
普通容器没锁
同步容器中锁的都是方法级别,也就是说锁的是整个容器,我们先来看下HashTable的锁
public synchronized V put(K key, V value) {}
public synchronized V remove(Object key) {}
可以看到:因为锁是内置锁,锁住的是整个容器
所以我们在put的时候,其他线程都不能put/get
而我们在get的时候,其他线程也都不能put/get
所以同步容器的效率会比较低
并发容器,我们以1.7的ConcurrentHashMap为例来说下(之所以选1.7,是因为它里面涉及的内容都是前面章节介绍过的)
它的锁粒度很小,它不会给整个容器上锁,而是分段上锁;
分段的依据就是key.hash,根据不同的hash值映射到不同的段(默认16个段),然后插入数据时,根据这个hash值去给对应的段上锁,此时其他段还是可以被其他线程读写的;
所以这就是文章开头所说的,为啥ConcurrentHashMap会支持多个线程同时写(因为只要插入的key的hashCode不会映射到同一个段里,那就不会冲突,此时就可以同时写)
读因为没有上锁,所以当然也支持同时读
如果读操作没有锁,那么它怎么保证数据的一致性呢?
答案就是以前介绍过的volatile(保证可见性、禁止重排序),它修饰在节点Node和值val上,保证了你get的值永远是最新的
下面是ConcurrentHashMap部分源码,可以看到val和net节点都是volatile类型
static class Node<K,V> implements Map.Entry<K,V> {
final int hash;
final K key;
volatile V val;
volatile Node<K,V> next;
}
总结下来就是:并发容器ConcurrentHashMap中,多个线程可同时读,多个线程可同时写,多个线程同时读和写
总结
- 什么是并发容器:并发容器是针对高并发专门设计的一些类,用来替代性能较低的同步容器
- 为什么会有并发容器:为了提高同步容器的性能
- 并发容器、同步容器、普通容器的区别:
- 性能:高 - 中 - 低
- 锁:粒度小 - 粒度大 - 无
- 场景:高并发 - 中并发 - 单线程
参考内容:
- 《Java并发编程实战》
- 《实战Java高并发》
- 《深入理解Java虚拟机》
后记
我这里介绍的都是比较浅的东西,其实并发容器的知识深入起来有很多;
但是因为这节是并发系列的比较靠前的,还有很多东西没涉及到,所以就分析地比较浅;
等到并发系列的内容都涉及地差不多了,再回过头来深入分析。
写在最后:
愿你的意中人亦是中意你之人。
Java并发容器篇的更多相关文章
- java 并发容器一之BoundedConcurrentHashMap(基于JDK1.8)
最近开始学习java并发容器,以补充自己在并发方面的知识,从源码上进行.如有不正确之处,还请各位大神批评指正. 前言: 本人个人理解,看一个类的源码要先从构造器入手,然后再看方法.下面看Bounded ...
- Java并发编程系列-(5) Java并发容器
5 并发容器 5.1 Hashtable.HashMap.TreeMap.HashSet.LinkedHashMap 在介绍并发容器之前,先分析下普通的容器,以及相应的实现,方便后续的对比. Hash ...
- Java 并发系列之六:java 并发容器(4个)
1. ConcurrentHashMap 2. ConcurrentLinkedQueue 3. ConcurrentSkipListMap 4. ConcurrentSkipListSet 5. t ...
- 《Java并发编程的艺术》第6/7/8章 Java并发容器与框架/13个原子操作/并发工具类
第6章 Java并发容器和框架 6.1 ConcurrentHashMap(线程安全的HashMap.锁分段技术) 6.1.1 为什么要使用ConcurrentHashMap 在并发编程中使用Has ...
- 【Java面试】- 并发容器篇
JDK 提供的并发容器 ConcurrentHashMap: 线程安全的 HashMap CopyOnWriteArrayList: 线程安全的 List,在读多写少的场合性能非常好,远远好于 Vec ...
- Java并发指南14:Java并发容器ConcurrentSkipListMap与CopyOnWriteArrayList
原文出处http://cmsblogs.com/ 『chenssy』 到目前为止,我们在Java世界里看到了两种实现key-value的数据结构:Hash.TreeMap,这两种数据结构各自都有着优缺 ...
- 【Java并发工具类】Java并发容器
前言 Java并发包有很大一部分都是关于并发容器的.Java在5.0版本之前线程安全的容器称之为同步容器.同步容器实现线程安全的方式:是将每个公有方法都使用synchronized修饰,保证每次只有一 ...
- java并发容器(Map、List、BlockingQueue)
转发: 大海巨浪 Java库本身就有多种线程安全的容器和同步工具,其中同步容器包括两部分:一个是Vector和Hashtable.另外还有JDK1.2中加入的同步包装类,这些类都是由Collectio ...
- java并发容器
同步容器将所有对容器状态的访问都串行化,以实现线程安全性.这种方式的缺点是严重降低并发性.Java 5.0提供了多种并发容器来改进同步容器的性能.如ConcurrentHashMap代替同步且基于散列 ...
随机推荐
- 再探命令行传参之c与python
继上一次java命令行传参 python sys模块包括了一组非常实用的服务,内含很多函数方法和变量,用来处理Python运行时配置以及资源,从而可以与前当程序之外的系统环境交互,如:python解释 ...
- 翻译:《实用的Python编程》04_03_Special_methods
目录 | 上一节 (4.2 继承) | 下一节 (4.4 异常) 4.3 特殊方法 可以通过特殊方法(或者称为"魔术"方法(magic method))自定义 Python 行为的 ...
- 选择 FreeBSD 而不是 Linux 的技术性原因1
Ports FreeBSD Ports 是一个惊人的工程壮举.NetBSD 的 pkgsrc (package source) 和 OpenBSD 的 ports collection 都源于 Fre ...
- 记一次scrapy-redis爬取小说网的分布式搭建过程
scrapy-redis简介 scrapy-redis是scrapy框架基于redis数据库的组件,用于scrapy项目的分布式开发和部署. 有如下特征: 分布式爬取 可以启动多个spider工程,相 ...
- Mysql之索引选择及优化
索引模型 哈希表 适用于只有等值查询的场景,Memory引擎默认索引 InnoDB支持自适应哈希索引,不可干预,由引擎自行决定是否创建 有序数组:在等值查询和范围查询场景中的性能都非常优秀,但插入和删 ...
- (二)SpringBoot启动过程的分析-环境信息准备
-- 以下内容均基于2.1.8.RELEASE版本 由上一篇SpringBoot基本启动过程的分析可以发现在run方法内部启动SpringBoot应用时采用多个步骤来实现,本文记录启动的第二个环节:环 ...
- C# - 实现类型的比较
IComparable<T> .NET 里,IComparable<T>是用来作比较的最常用接口. 如果某个类型的实例需要与该类型的其它实例进行比较或者排序的话,那么该类型就可 ...
- var=value?export前后差在哪?-- Shell十三问<第五问>
var=value?export前后差在哪?-- Shell十三问<第五问> 这次让我们暂时丢开 command line ,先来了解一下 bash 变量(variable)吧.所谓的 变 ...
- Spring Cloud 升级之路 - 2020.0.x - 3. Undertow 的 accesslog 配置
上一节我们讲述了如何使用 Undertow 作为我们的 Web 服务容器,本小节我们来分析使用 Undertow 的另一个问题,也就是如何配置 accesslog,以及 accesslog 的各种占位 ...
- [Fundamental of Power Electronics]-PART I-2.稳态变换器原理分析-2.5/2.6 多极点滤波器电压纹波估计及要点小结
2.5 含两极点低通滤波器变换器的输出电压纹波估计 在分析包含两极点低通滤波器的变换器如Cuk变换器及Buck变换器(图2.25)输出时,小纹波近似将会失效.对于这些变换器而言,无论输出滤波电容的值是 ...