1. 是什么

写时复制的set,有序不重复,底层使用CopyOnWriteArrayList实现

2. 如何使用

public class CopyOnWriteArraySetTest
{
public static void main(String[] args)
{
CopyOnWriteArraySet<String> set = new CopyOnWriteArraySet<>(); set.add("a");
set.add("b");
set.add("c"); set.remove("a");
System.out.println(set.contains("a"));//false System.out.println(Thread.currentThread().getName() + "读取set:" + set);//[b, c] new Thread(()->{
System.out.println(Thread.currentThread().getName() + "读取set:" + set);//[c]
try
{
TimeUnit.SECONDS.sleep(2);
}
catch (InterruptedException e)
{
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName() + "再次读取set:" + set);//[c] }).start();
new Thread(()->{
System.out.println(Thread.currentThread().getName() + "读取set:" + set);//[c]
try
{
TimeUnit.SECONDS.sleep(2);
}
catch (InterruptedException e)
{
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName() + "再次读取set:" + set);//[c]
}).start(); try
{
TimeUnit.SECONDS.sleep(1);
}
catch (InterruptedException e)
{
e.printStackTrace();
}
set.remove("b"); System.out.println(Thread.currentThread().getName() + "读取修改后的set:" + set);//[c] }
}

3. 原理分析

3.1. 构造方法

3.1.1. 底层使用CopyOnWriteArrayList实现

public class CopyOnWriteArraySet<E> extends AbstractSet<E>
implements java.io.Serializable {
//底层使用的是CopyOnWriteArrayList实现
private final CopyOnWriteArrayList<E> al; public CopyOnWriteArraySet() {
//默认一个元素的数组
al = new CopyOnWriteArrayList<E>();
}
}

3.2. add方法

public boolean add(E e) {
//调用CopyOnWriteArrayList addIfAbsent方法
return al.addIfAbsent(e);
}

3.2.1. 转调CopyOnWriteArrayList addIfAbsent

  • CopyOnWriteArrayList addIfAbsent
public boolean addIfAbsent(E e) {
//获取原数组
Object[] snapshot = getArray();
//如果在数组中已经存在,那么返回false
return indexOf(e, snapshot, 0, snapshot.length) >= 0 ? false :
//元素还不在数组中,则加入
addIfAbsent(e, snapshot);
}
  • 3行:获取原数组
  • 5行:遍历数组是否存在该object
  • 7行:不在的话插入list尾部

下面分别说明:

3.2.1.1. 遍历数组是否存在该object
private static int indexOf(Object o, Object[] elements,
int index, int fence) {
//object为null
if (o == null) {
for (int i = index; i < fence; i++)
if (elements[i] == null)
return i;
//object不为null
} else {
for (int i = index; i < fence; i++)
if (o.equals(elements[i]))
return i;
}
return -1;
}
3.2.1.2. 不在的话则加锁插入list尾部
  • CopyOnWriteArrayList addIfAbsent
//把e加入到snapshot数组中
private boolean addIfAbsent(E e, Object[] snapshot) {
final ReentrantLock lock = this.lock;
lock.lock();
try {
Object[] current = getArray();
int len = current.length;
//原数组跟现在的数组不一样了
if (snapshot != current) {
// Optimize for lost race to another addXXX operation
int common = Math.min(snapshot.length, len);
//遍历改变后的数组左边
for (int i = 0; i < common; i++)
//有一个位置不相同 且 改变后的数组这个位置xx的元素与要添加的元素相同,说明已经存在,返回false
if (current[i] != snapshot[i] && eq(e, current[i]))
return false;
//从改变后的数组右边查看是否存在要添加的元素
if (indexOf(e, current, common, len) >= 0)
return false;
}
//复制原数组并扩容,把e加入到数组的末尾,set回array
Object[] newElements = Arrays.copyOf(current, len + 1);
newElements[len] = e;
setArray(newElements);
return true;
} finally {
lock.unlock();
}
}

大体逻辑跟10.CopyOnWriteArrayList.md的remove方法差不多

3.3. contains方法

public boolean contains(Object o) {
//转调CopyOnWriteArrayList的contains方法
return al.contains(o);
}

3.3.1. 转调CopyOnWriteArrayList的contains

  • CopyOnWriteArrayList的contains
//CopyOnWriteArrayList的contains方法
public boolean contains(Object o) {
Object[] elements = getArray();
//遍历数组查找
return indexOf(o, elements, 0, elements.length) >= 0;
}
3.3.1.1. 遍历数组判断是否存在该object
private static int indexOf(Object o, Object[] elements,
int index, int fence) {
//object为null
if (o == null) {
for (int i = index; i < fence; i++)
if (elements[i] == null)
return i;
//object不为null
} else {
for (int i = index; i < fence; i++)
if (o.equals(elements[i]))
return i;
}
return -1;
}

3.4. remove方法

  • remove
public boolean remove(Object o) {
//转调CopyOnWriteArrayList的remove方法
return al.remove(o);
}

3.4.1. 转调CopyOnWriteArrayList的remove方法

  • CopyOnWriteArrayList remove
//CopyOnWriteArrayList的remove方法
public boolean remove(Object o) {
Object[] snapshot = getArray();
int index = indexOf(o, snapshot, 0, snapshot.length);
return (index < 0) ? false : remove(o, snapshot, index);
}
  • 3行:获取原数组
  • 4行:遍历数组是否存在该object
  • 5行:存在的话调用CopyOnWriteArrayList的remove方法删除
3.4.1.1. 遍历底层数组判断是否存在该object
private static int indexOf(Object o, Object[] elements,
int index, int fence) {
//object为null
if (o == null) {
for (int i = index; i < fence; i++)
if (elements[i] == null)
return i;
//object不为null
} else {
for (int i = index; i < fence; i++)
if (o.equals(elements[i]))
return i;
}
return -1;
}
3.4.1.2. 存在的话再去删除

参考10.CopyOnWriteArrayList.md remove方法

4. 总结

底层直接通过调用CopyOnWriteArrayList实现的,因此有序

5. 参考

Java源码分析系列笔记-11.CopyOnWriteArraySet的更多相关文章

  1. Java源码分析系列之HttpServletRequest源码分析

    从源码当中 我们可以 得知,HttpServletRequest其实 实际上 并 不是一个类,它只是一个标准,一个 接口而已,它的 父类是ServletRequest. 认证方式 public int ...

  2. Java源码分析系列

    1) 深入Java集合学习系列:HashMap的实现原理 2) 深入Java集合学习系列:LinkedHashMap的实现原理 3) 深入Java集合学习系列:HashSet的实现原理 4) 深入Ja ...

  3. spring源码分析系列 (8) FactoryBean工厂类机制

    更多文章点击--spring源码分析系列 1.FactoryBean设计目的以及使用 2.FactoryBean工厂类机制运行机制分析 1.FactoryBean设计目的以及使用 FactoryBea ...

  4. Spring mvc源码分析系列--Servlet的前世今生

    Spring mvc源码分析系列--Servlet的前世今生 概述 上一篇文章Spring mvc源码分析系列--前言挖了坑,但是由于最近需求繁忙,一直没有时间填坑.今天暂且来填一个小坑,这篇文章我们 ...

  5. jQuery源码分析系列

    声明:本文为原创文章,如需转载,请注明来源并保留原文链接Aaron,谢谢! 版本截止到2013.8.24 jQuery官方发布最新的的2.0.3为准 附上每一章的源码注释分析 :https://git ...

  6. MyCat源码分析系列之——结果合并

    更多MyCat源码分析,请戳MyCat源码分析系列 结果合并 在SQL下发流程和前后端验证流程中介绍过,通过用户验证的后端连接绑定的NIOHandler是MySQLConnectionHandler实 ...

  7. MyCat源码分析系列之——BufferPool与缓存机制

    更多MyCat源码分析,请戳MyCat源码分析系列 BufferPool MyCat的缓冲区采用的是java.nio.ByteBuffer,由BufferPool类统一管理,相关的设置在SystemC ...

  8. jquery2源码分析系列

    学习jquery的源码对于提高前端的能力很有帮助,下面的系列是我在网上看到的对jquery2的源码的分析.等有时间了好好研究下.我们知道jquery2开始就不支持IE6-8了,从jquery2的源码中 ...

  9. [Tomcat 源码分析系列] (二) : Tomcat 启动脚本-catalina.bat

    概述 Tomcat 的三个最重要的启动脚本: startup.bat catalina.bat setclasspath.bat 上一篇咱们分析了 startup.bat 脚本 这一篇咱们来分析 ca ...

  10. [转]jQuery源码分析系列

    文章转自:jQuery源码分析系列-Aaron 版本截止到2013.8.24 jQuery官方发布最新的的2.0.3为准 附上每一章的源码注释分析 :https://github.com/JsAaro ...

随机推荐

  1. Delphi 中拖动无边框窗口的5种方法

    1.MouseMove事件中加入: // ReleaseCapture; // Perform(WM_SYSCOMMAND, $F017 , 0); 2.MouseDown事件中加入: // POST ...

  2. delphi 让执行程序不在任务栏显示

    Application.MainFormOnTaskbar := False; procedure TForm1.FormShow(Sender: TObject); begin ShowWindow ...

  3. 保姆级教程——手把手教会你如何在Linux上安装Redis

    一.Linux系统安装Redis(7.4.0) 注意: 全程是在root底下操作,当然也可以采用sudo 1.1 安装Redis依赖 Redis是基于C语言编写的,因此首先需要安装Redis所需要的g ...

  4. 再谈MCP协议,看看 MCP 是如何重塑 AI 与外部数据源互动的能力?

    Techscribe Central 缩略图由 Techscribe Central 制作和编辑 MCP!!是不是一头雾水?我当时也是这个反应.我也是最近才听说它开始引发关注,然后我发现大多数人根本不 ...

  5. CSAPP学习笔记——chapter8 异常控制流

    CSAPP学习笔记--chapter8 异常控制流 简介 异常控制流(Exceptional Control Flow,ECF)是在计算机系统中处理不寻常或异常情况的一种机制.它允许系统跳出正常的顺序 ...

  6. study Rust-4【所有权】这个太重要了!

    由于Rust内存垃圾自动回收,那就得搞清楚这个所有权玩意.这个太重要了.因为关系到贯穿于你以后的程序编写. 几个概念: 一.移动 1.咱们一般语言,自己申请内存,自己管理和释放.就是new和free. ...

  7. 邮件自动回复助手(Rasa/SMTP)实现教程

    在现代办公场景中,处理大量邮件是一项既耗时又容易出错的任务.为了提升工作效率,我们可以利用自然语言处理(NLP)和邮件传输协议(SMTP)技术,构建一个智能的邮件自动回复助手.本文将详细介绍如何使用P ...

  8. Docker自定义镜像输出日志

    概述 本文主要解决Docker自定义镜像之后,通过docker logs命令查看不到相关日志的问题 在 Docker 中自定义镜像输出日志,通常需要确保你的应用程序将日志输出到 标准输出(stdout ...

  9. 数据库事务隔离与Alembic数据恢复的实战艺术

    title: 数据库事务隔离与Alembic数据恢复的实战艺术 date: 2025/05/15 00:05:13 updated: 2025/05/15 00:05:13 author: cmdra ...

  10. 【网站推荐】推荐几个MCP(Model Context Protocol,模型上下文协议)的网站

    MCP网站一:MCP 快速入门 介绍 MCP 快速入门.开发工具.核心架构等内容. https://mcp-docs.cn/introduction MCP网站二:GitHub 官方仓库 由MCP协议 ...