并发容器-CopyOnWriteArrayList

1 /** The array, accessed only via getArray/setArray. */
2 private transient volatile Object[] array;
1 /**
2 * {@inheritDoc}
3 *
4 * @throws IndexOutOfBoundsException {@inheritDoc}
5 */
6 public E get(int index) {
7 return get(getArray(), index);
8 }
9 /**
10 * Gets the array. Non-private so as to also be accessible
11 * from CopyOnWriteArraySet class.
12 */
13 final Object[] getArray() {
14 return array;
15 }
16 @SuppressWarnings("unchecked")
17 private E get(Object[] a, int index) {
18 return (E) a[index];
19 }
1 /**
2 * Replaces the element at the specified position in this list with the
3 * specified element.
4 *
5 * @throws IndexOutOfBoundsException {@inheritDoc}
6 */
7 public E set(int index, E element) {
8 final ReentrantLock lock = this.lock;
9 lock.lock();
10 try {
11 Object[] elements = getArray();
12 E oldValue = get(elements, index);
13 // 判断原有位置元素是否和插入元素相同,不相同才插入
14 if (oldValue != element) {
15 int len = elements.length;
16 Object[] newElements = Arrays.copyOf(elements, len);
17 newElements[index] = element;
18 setArray(newElements);
19 } else {
20 // Not quite a no-op; ensures volatile write semantics
21 // 不是无用操作,是为了保证volatile写的意义
22 setArray(elements);
23 }
24 return oldValue;
25 } finally {
26 lock.unlock();
27 }
28 }
29 /**
30 * Sets the array.
31 */
32 final void setArray(Object[] a) {
33 array = a;
34 }
// initial conditions
int nonVolatileField = 0;
CopyOnWriteArrayList<String> list = /* a single String */ // Thread 1
nonVolatileField = 1; // (1)
list.set(0, "x"); // (2) // Thread 2
String s = list.get(0); // (3)
if (s == "x") {
int localVar = nonVolatileField; // (4)
}
1 /**
2 * Appends the specified element to the end of this list.
3 * 末尾增加元素
4 * @param e element to be appended to this list
5 * @return {@code true} (as specified by {@link Collection#add})
6 */
7 public boolean add(E e) {
8 final ReentrantLock lock = this.lock;
9 lock.lock();
10 try {
11 Object[] elements = getArray();
12 int len = elements.length;
13 Object[] newElements = Arrays.copyOf(elements, len + 1);
14 newElements[len] = e;
15 setArray(newElements);
16 return true;
17 } finally {
18 lock.unlock();
19 }
20 }
21 /**
22 * Inserts the specified element at the specified position in this
23 * list. Shifts the element currently at that position (if any) and
24 * any subsequent elements to the right (adds one to their indices).
25 *
26 * @throws IndexOutOfBoundsException {@inheritDoc}
27 */
28 public void add(int index, E element) {
29 final ReentrantLock lock = this.lock;
30 lock.lock();
31 try {
32 Object[] elements = getArray();
33 int len = elements.length;
34 if (index > len || index < 0)
35 throw new IndexOutOfBoundsException("Index: "+index+
36 ", Size: "+len);
37 Object[] newElements;
38 int numMoved = len - index;
39 if (numMoved == 0)
40 newElements = Arrays.copyOf(elements, len + 1);
41 else {
42 newElements = new Object[len + 1];
43 System.arraycopy(elements, 0, newElements, 0, index);
44 System.arraycopy(elements, index, newElements, index + 1,
45 numMoved);
46 }
47 newElements[index] = element;
48 setArray(newElements);
49 } finally {
50 lock.unlock();
51 }
52 }
可以看到,增加元素和更新元素一样,也是创建一个新数组复制原有元素,然后插入新增元素。同时也是加了可重入锁,同一时刻只有一个线程能进行修改。
/**
* Removes the element at the specified position in this list.
* Shifts any subsequent elements to the left (subtracts one from their
* indices). Returns the element that was removed from the list.
*
* @throws IndexOutOfBoundsException {@inheritDoc}
*/
public E remove(int index) {
final ReentrantLock lock = this.lock;
lock.lock();
try {
Object[] elements = getArray();
int len = elements.length;
E oldValue = get(elements, index);
int numMoved = len - index - 1;
if (numMoved == 0)
setArray(Arrays.copyOf(elements, len - 1));
else {
Object[] newElements = new Object[len - 1];
System.arraycopy(elements, 0, newElements, 0, index);
System.arraycopy(elements, index + 1, newElements, index,
numMoved);
setArray(newElements);
}
return oldValue;
} finally {
lock.unlock();
}
}
并发容器-CopyOnWriteArrayList的更多相关文章
- Java并发编程原理与实战三十四:并发容器CopyOnWriteArrayList原理与使用
1.ArrayList的实现原理是怎样的呢? ------>例如:ArrayList本质是实现了一个可变长度的数组. 假如这个数组的长度为10,调用add方法的时候,下标会移动到下一位,当移动到 ...
- 多线程并发容器CopyOnWriteArrayList
原文链接: http://ifeve.com/java-copy-on-write/ Copy-On-Write简称COW,是一种用于程序设计中的优化策略.其基本思路是,从一开始大家都在共享同一个内容 ...
- Java并发容器——CopyOnWriteArrayList
CopyOnWriteArrayList是“读写分离”的容器,在写的时候是先将底层源数组复制到新数组中,然后在新数组中写,写完后更新源数组.而读只是在源数组上读.也就是,读和写是分离的.由于,写的时候 ...
- Java并发机制(5)--同步容器与并发容器
Java并发编程:同步容器整理自:博客园-海子-http://www.cnblogs.com/dolphin0520/p/3933404.html1.同步容器出现原因 常用的ArrayList,Lin ...
- JAVA 多线程随笔 (三) 多线程用到的并发容器 (ConcurrentHashMap,CopyOnWriteArrayList, CopyOnWriteArraySet)
1.引言 在多线程的环境中,如果想要使用容器类,就需要注意所使用的容器类是否是线程安全的.在最早开始,人们一般都在使用同步容器(Vector,HashTable),其基本的原理,就是针对容器的每一个操 ...
- CopyOnWriteArrayList并发容器
CopyOnWriteArrayList并发容器 Copy-On-Write简称COW,是一种用于程序设计中的优化策略.其基本思路是,从一开始大家都在共享同一个内容,当某个人想要修改这个内容的时候,才 ...
- 09 jdk1.5的并发容器:CopyOnWriteArrayList(转载)
原文链接:http://ifeve.com/java-copy-on-write/ Copy-On-Write简称COW,是一种用于程序设计中的优化策略. 其基本思路是,从一开始大家都在共享同一个内容 ...
- 【java并发容器】并发容器之CopyOnWriteArrayList
原文链接: http://ifeve.com/java-copy-on-write/ Copy-On-Write简称COW,是一种用于程序设计中的优化策略.其基本思路是,从一开始大家都在共享同一个内容 ...
- Java并发指南14:Java并发容器ConcurrentSkipListMap与CopyOnWriteArrayList
原文出处http://cmsblogs.com/ 『chenssy』 到目前为止,我们在Java世界里看到了两种实现key-value的数据结构:Hash.TreeMap,这两种数据结构各自都有着优缺 ...
随机推荐
- .NET并发编程-任务函数并行
本系列学习在.NET中的并发并行编程模式,实战技巧 请问普通: 被门夹过的核桃还能补脑吗 本小节开始学习基于任务的函数式并行.本系列保证最少代码呈现量,虽然talk is cheap, show me ...
- Go语言学习 学习资料汇总
从进入实验室以来,一直听小溪师兄说Go语言,但是第一学期的课很多,一直没有时间学习,现在终于空出来时间学习,按照我的学习习惯,我一般分为三步走 学习一门语言首先要知道学会了能干什么, 然后再把网上的资 ...
- 在ASP.NET Core中用HttpClient(三)——发送HTTP PATCH请求
在前面的两篇文章中,我们讨论了很多关于使用HttpClient进行CRUD操作的基础知识.如果你已经读过它们,你就知道如何使用HttpClient从API中获取数据,并使用HttpClient发送PO ...
- Paperfolding HDU - 6822
传送门:https://vjudge.net/problem/HDU-6822 题意:给你一张无限的纸有四种折叠方式,并且在n次折叠后减两刀问最后纸张数量的数学期望. 思路:我们要得到一个通项公式对于 ...
- Linux入门视频笔记一(基本命令)
一.简单命令 1.date:当前时间 2.cal:当前日期(日历格式) ①cal 2019:2019年全年日历 ②cal 1 2019:2019年1月份 二.Linux文件结构 1.根目录:root( ...
- 初识Django(一)
首先安装Django 1 pip install django==1.11.13 安装 由于django最新的长期支持版本为1.11.x,所以我们安装最新的1.11.13版本 '=='后面跟版本号 安 ...
- Python基础之数据类型详解(2)
今天继续昨天的python基本数据类型详解,按照上一篇博文的格式,接下来讲解列表.元组.字典以及集合. 列表 1.用途按位置存放多个值2.定义在[]内用逗号分割开多个任意类型的元素 # 定义列表 # ...
- Java例题_19 打印菱形图案
1 /*19 [程序 19 打印菱形图案] 2 题目:打印出如下图案(菱形) 3 * 4 *** 5 ***** 6 ******* 7 ***** 8 *** 9 * 10 */ 11 12 /*分 ...
- Java例题_39 判断奇偶后分数累加
1 /*39 [程序 39 分数累加] 2 题目:编写一个函数,输入 n 为偶数时,调用函数求 1/2+1/4+...+1/n,当输入 n 为奇数时,调用函数 3 1/1+1/3+...+1/n 4 ...
- IT培训有哪些坑(二)?
今天继续给大家分享一下IT培训都有哪些坑?有哪些不靠谱? 做招转的不靠谱.什么是招转?就是招聘转招生,名义上说的是招聘,但实际上做的就是招生.有很多大学刚毕业的计算机相关专业的同学,他们大学毕业之后, ...