List的同步类比较
TL;NRs
CopyOnWriteArrayList类在多线程顺序读取上有很大的优势,但在随机读取上反而有较大的劣势,且在写入方面性能极差。Vector类在顺序读取方面性能较差,但在随机读取方面有较大的优势,写入方面性能也还可以。
1,引言
java线程安全的List实现有以下三种:
new Vector<>()Collections.synchronizedList(new ArrayList<>())new CopyOnWriteArrayList<>()
通常认为使用了synchronized会导致运行变慢,那么在java针对synchronized进行一系列优化后,现在的情况如何呢?为了检验这一说法,写了一个验证程序进行验证。
2,验证代码
以ArrayList作为基础,分别测试4种List的顺序写入(0 ~ 1 << 24)、顺序读取和随机读取,各十轮。据此编写代码。代码太长了,所以放到最后
3,测试平台
垃圾笔记本,使用Intel酷睿i5 7200U
java版本为java 12,HotSpot虚拟机
4,测试结果
单位:毫秒

5,结果分析
ArrayList(A)、Vector(V)、Collections.synchronizedList(new ArrayList<>())(S)、以及CopyOnWriteArrayList(C)四种类型的结果分别如下
十轮写入,单位毫秒
| A | V | S | C | |
|---|---|---|---|---|
| 总时间 | 6426 | 9365 | 10186 | inf |
| 最大时间 | 1313 | 1016 | 1096 | inf |
| 最小时间 | 239 | 815 | 672 | inf |
十轮单线程顺序读,单位毫秒
| A | V | S | C | |
|---|---|---|---|---|
| 总时间 | 41 | 2247 | 1538 | 1560 |
| 最大时间 | 22 | 418 | 200 | 167 |
| 最小时间 | 0 | 196 | 129 | 148 |
十轮单线程随机读,单位毫秒
| A | V | S | C | |
|---|---|---|---|---|
| 总时间 | 2167 | 4908 | 11792 | 11133 |
| 最大时间 | 256 | 573 | 1372 | 1264 |
| 最小时间 | 202 | 473 | 1110 | 1030 |
十线程顺序读,单位毫秒
| V | S | C | |
|---|---|---|---|
| 总时间 | 11232 | 12650 | 696 |
十线程随机读,单位毫秒
| V | S | C | |
|---|---|---|---|
| 总时间 | 16828 | 17888 | 26089 |
6,结论
单线程写入性能:A > V = S >>>> C
单线程顺序读取性能:A >> S = C > V
单线程随机读取性能:A > V > S = C
20线程顺序读取性能:C >> V > S
20线程随机读取性能:V > S >> C
COW顺序读取性能较好,随机读取性能较差,写入性能极差。
Vector随机读取性能较好,顺序读取性能和写入性能较差。
附录 测试代码
import java.util.*;
import java.util.concurrent.*;
public class VectorTest {
private static final int CNT = 1 << 24;
private static final Random rand = new Random();
public static void main(String[] args) throws InterruptedException {
int writeRound = 10, readRound = 10, randomReadRound = 10;
int nRead = 20, nRandomRead = 20;
List<Integer> lsA = new ArrayList<>();
List<Integer> lsV = new Vector<>();
List<Integer> lsS = Collections.synchronizedList(new ArrayList<>());
List<Integer> lsC = new CopyOnWriteArrayList<>();
test(lsA, "ArrayList", writeRound, readRound, randomReadRound);
test(lsV, "Vector", writeRound, readRound, randomReadRound);
test(lsS, "SynArrayList", writeRound, readRound, randomReadRound);
lsC.addAll(lsA);
test(lsC, "COWList", 0, readRound, randomReadRound);
multiThreadTest(lsV, "Vector", nRead, nRandomRead);
multiThreadTest(lsS, "SynArrayList", nRead, nRandomRead);
multiThreadTest(lsC, "COWList", nRead, nRandomRead);
}
private static void test(List<Integer> list, String name, int writeRound, int readRound, int randomReadRound) {
int max = 0, min = Integer.MAX_VALUE, sum = 0;
int[] w = new int[writeRound], r = new int[readRound], rr = new int[randomReadRound];
for (int i = 0; i < writeRound; i++) {
list.clear();
int v = w[i] = writeTest(list);
max = Math.max(max, v);
min = Math.min(min, v);
sum += v;
}
System.out.printf("%s write test: sum = %d, max = %d, min = %d\n", name, sum, max, min);
for (int v : w) System.out.printf("%d\t", v);
System.out.println();
sum = max = 0;
min = Integer.MAX_VALUE;
for (int i = 0; i < readRound; i++) {
int v = r[i] = readTest(list);
max = Math.max(max, v);
min = Math.min(min, v);
sum += v;
}
System.out.printf("%s read test: sum = %d, max = %d, min = %d\n", name, sum, max, min);
for (int v : r) System.out.printf("%d\t", v);
System.out.println();
sum = max = 0;
min = Integer.MAX_VALUE;
for (int i = 0; i < randomReadRound; i++) {
int v = rr[i] = randomReadTest(list);
max = Math.max(max, v);
min = Math.min(min, v);
sum += v;
}
System.out.printf("%s random read test: sum = %d, max = %d, min = %d\n", name, sum, max, min);
for (int v : rr) System.out.printf("%d\t", v);
System.out.println();
}
private static int writeTest(List<Integer> list) {
long t0 = System.currentTimeMillis();
for (int i = 0; i < CNT; i++) list.add(i);
long t1 = System.currentTimeMillis();
return (int)(t1 - t0);
}
private static int readTest(List<Integer> list) {
long t0 = System.currentTimeMillis();
for (int i = 0; i < CNT; i++) list.get(i);
long t1 = System.currentTimeMillis();
return (int)(t1 - t0);
}
private static int randomReadTest(List<Integer> list) {
long t0 = System.currentTimeMillis();
for (int i = 0; i < CNT; i++) list.get(rand.nextInt(CNT));
long t1 = System.currentTimeMillis();
return (int)(t1 - t0);
}
private static List<Integer> ls;
private static long t2 = 0;
private static CountDownLatch cdl;
public static class ThreadRead extends Thread {
public void run() {
for (int i = 0; i < CNT; i++) ls.get(i);
long t1 = System.currentTimeMillis();
t2 = Math.max(t1, t2);
cdl.countDown();
}
}
public static class ThreadRandomRead extends Thread {
public void run() {
for (int i = 0; i < CNT; i++) ls.get(rand.nextInt(CNT));
long t1 = System.currentTimeMillis();
t2 = Math.max(t1, t2);
cdl.countDown();
}
}
private static void multiThreadTest(List<Integer> list, String name, int nRead, int nRandomRead) throws InterruptedException {
int tr = 0, trr = 0;
ls = list;
cdl = new CountDownLatch(nRead);
long t0 = System.currentTimeMillis();
for (int i = 0; i < nRead; i++) {
new ThreadRead().start();
}
cdl.await();
tr = (int)(t2 - t0);
cdl = new CountDownLatch(nRandomRead);
t2 = 0;
t0 = System.currentTimeMillis();
for (int i = 0; i < nRandomRead; i++) {
new ThreadRandomRead().start();
}
cdl.await();
trr = (int)(t2 - t0);
System.out.printf("%s: tr = %d, trr = %d\n", name, tr, trr);
}
}
List的同步类比较的更多相关文章
- Java中多线程同步类 CountDownLatch
在多线程开发中,常常遇到希望一组线程完成之后在执行之后的操作,java提供了一个多线程同步辅助类,可以完成此类需求: 类中常见的方法: 其中构造方法:CountDownLatch(int count) ...
- 16.同步类容器Collections.synchronized
voctor动态数组.同步类容器,底层实现基于:Collections.synchronized package demo5; import java.util.ArrayList; import j ...
- 15.同步类容器Vector
同步类容器1 1.线程都是安全的. 2.在某些场景下需要加锁来保护“复合操作” a.迭代:反复去访问元素.遍历完容器所有的元素 b.跳转:根据下标制定去访问查找元素 c.条件运算 3.复合操作在多线程 ...
- Java线程同步类容器和并发容器(四)
同步类容器都是线程安全的,在某些场景下,需要枷锁保护符合操作,最经典ConcurrentModifiicationException,原因是当容器迭代的过程中,被并发的修改了内容. for (Iter ...
- 同步类容器和并发类容器——ConcurrentMap、CopyOnWrite、Queue
一 同步类容器同步类容器都是线程安全的,但在某些场景中可能需要加锁来保证复合操作. 符合操作如:迭代(反复访问元素,遍历完容器中所有元素).跳转(根据指定的顺序找到当前元素的下一个元素).条件运算. ...
- Java多线程信号量同步类CountDownLatch与Semaphore
信号量同步是指在不同线程之间,通过传递同步信号量来协调线程执行的先后次序.CountDownLatch是基于时间维度的Semaphore则是基于信号维度的. 1:基于执行时间的同步类CountDown ...
- 同步类的基础AbstractQueuedSynchronizer(AQS)
同步类的基础AbstractQueuedSynchronizer(AQS) 我们之前介绍了很多同步类,比如ReentrantLock,Semaphore, CountDownLatch, Reentr ...
- 解读java同步类CountDownLatch
同步辅助类: CountDownLatch是一个同步辅助类,在jdk5中引入,它允许一个或多个线程等待其他线程操作完成之后才执行. 实现原理 : CountDownLatch是通过计数器的方式来实现, ...
- synchronized 线程同步-类级别锁定
1.demo 说明:代码中通过 printNum 方法传入参数判断 a.b 分别对 num 这个参数的值进行了修改. package demo1; import sun.applet.Main; pu ...
随机推荐
- 判断数据类型(typeof&instanceof&toString)
一.数据类型 ES6规范中有7种数据类型,分别是基本类型和引用类型两大类 基本类型(简单类型.原始类型):String.Number.Boolean.Null.Undefined.Symbol 引用类 ...
- 2021年第十二届蓝桥杯javaA组省赛部分题目
试题 D: 路径 本题总分:10 分 [问题描述] 小蓝学习了最短路径之后特别高兴,他定义了一个特别的图,希望找到图 中的最短路径. 小蓝的图由 2021 个结点组成,依次编号 1 至 2021. 对 ...
- Java的标识符与关键字
目录 Java关键字 总表:java关键字共53个(其中包含两个保留字const,goto) Java标识符 定义 组成 命名规则 视频课程 Java关键字 Java关键字是电脑语言里事先定义的,有特 ...
- Navicat 连接 MySQL
目录 简述 新建连接 常见错误 简述 Navicat 是一套快速.可靠和全面的数据库管理工具,专门用于简化数据库管理和降低管理成本.Navicat 图形界面直观,提供简便的管理方法,设计和操作 MyS ...
- 如何在 Mac 和虚拟机上安装 macOS Big Sur、Monterey 和 Ventura
请访问原文链接:https://sysin.org/blog/how-to-install-macos/,查看最新版.原创作品,转载请保留出处. 作者主页:www.sysin.org 名词解释: 硬件 ...
- 聊聊 RPA 方向的规划:简单有价值的事情长期坚持做
「简单有价值的事情长期坚持做」 这是成功最简单,但也最难学的秘诀.不经过训练,人很难意识到时间复利的威力. 仙剑奇侠传的「十里坡剑神」和金庸群侠传的「十级野球拳」,就是简单的事情持之以恒反复做,最后就 ...
- Stream常用操作以及原理探索
Stream常用操作以及原理 Stream是什么? Stream是一个高级迭代器,它不是数据结构,不能存储数据.它可以用来实现内部迭代,内部迭代相比平常的外部迭代,它可以实现并行求值(高效,外部迭代要 ...
- Spring IOC 为什么能降低耦合
有同学在学习 Spring 框架中可能会问这样的问题,为什么通过依赖注入就可以降低代码间的耦合呢?我通过 new 生产对象不也可以吗,不就是一行代码的不同,一个是 @Resource 注入,一个是 n ...
- linux系统调优工具
系统调优思路 性能优化就是找到系统处理中的瓶颈以及去除这些的过程,性能优化其实是对 OS 各子系统达到一种平衡的定义.具体步骤如下: 1. 系统的运行状况: CPU -> MEM -> D ...
- NC204382 中序序列
NC204382 中序序列 题目 题目描述 给定一棵有 \(n\) 个结点的二叉树的先序遍历与后序遍历序列,求其中序遍历序列. 若某节点只有一个子结点,则此处将其看作左儿子结点 示例1 输入 5,[3 ...