我的编程空间,编程开发者的网络收藏夹
学习永远不晚

Java的CopyOnWrite怎么实现

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

Java的CopyOnWrite怎么实现

这篇文章主要介绍了Java的CopyOnWrite怎么实现的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇Java的CopyOnWrite怎么实现文章都会有所收获,下面我们一起来看看吧。

概念

CopyOnWrite 是什么呢,从字面上看,就是在写入时复制。看起来貌似很简单,那么写入时复制,具体是怎么实现的呢?

先来说说思想,具体怎么实现等下分析

CopyOnWrite 的思想就是:当向一个容器中添加元素的时候,不是直接在当前这个容器里面添加的,而是复制出来一个新的容器,在新的容器里面添加元素,添加完毕之后再将原容器的引用指向新的容器,这样就实现了写入时复制

你还记得在提到数据库的时候,一般都会说主从复制,读写分离吗?CopyOnWrite 的设计思想是不是和经常说的主从复制,读写分离如出一撤?

优缺点

了解概念之后,对它的优缺点应该就比较好理解了

优点就是,读和写可以并行执行,因为读的是原来的容器,写的是新的容器,它们之间互不影响,所以读和写是可以并行执行的,在某些高并发场景下,可以提高程序的响应时间

但是呢,你也看到了, CopyOnWrite 是在写入的时候,复制了一个新的容器出来,所以要考虑它的内存开销问题,又回到了在学算法时一直强调的一个思想:拿空间换时间

需要注意一下,它只保证数据的最终一致性。因为在读的时候,读取的内容是原容器里面的内容,新添加的内容是读取不到的

基于它的优缺点应该就可以得出一个结论:CopyOnWrite 适用于写操作非常少的场景,而且还能够容忍读写的暂时不一致 如果你的应用场景不适合,那还是考虑使用别的方法来实现吧

还有一点需要注意的是:在写入时,它会复制一个新的容器,所以如果有写入需求的话,最好可以批量写入,因为每次写入的时候,容器都会进行复制,如果能够减少写入的次数,就可以减少容器的复制次数

JUC 包下,实现 CopyOnWrite 思想的就是 CopyOnWriteArrayList & CopyOnWriteArraySet 这两个方法,本篇文章侧重于讲清楚 CopyOnWriteArrayList

CopyOnWriteArrayList

CopyOnWriteArrayList 中,需要注意的是 add 方法:

    public boolean add(E e) {        final ReentrantLock lock = this.lock;        // 在写入的时候,需要加锁,如果不加锁的话,在多线程场景下可能会被 copy 出 n 个副本出来        // 加锁之后,就能保证在进行写时,只有一个线程在操作        lock.lock();        try {            Object[] elements = getArray();            int len = elements.length;            // 复制原来的数组            Object[] newElements = Arrays.copyOf(elements, len + 1);            // 将要添加的元素添加到新数组中            newElements[len] = e;            // 将对原数组的引用指向新的数组            setArray(newElements);            return true;        } finally {            lock.unlock();        }    }

在写的时候需要加锁,但是在读取的时候不需要添加

因为读取的是原数组的元素,对新数组没有什么影响,加了锁反而会增加性能开销

public E get(int index) { return get(getArray(), index);}

举个例子:

CopyOnWriteJUC 包下,那么它就保证了线程安全

咱们来做个小 demo 验证一下:

@Slf4jpublic class ArrayListExample {    // 请求总数    public static int clientTotal = 5000;    // 同时并发执行的线程数    public static int threadTotal = 200;    private static List<Integer> list = new ArrayList<>();    public static void  main(String[] args) throws Exception{        ExecutorService executorService = Executors.newCachedThreadPool();        final Semaphore semaphore = new Semaphore(threadTotal);        final CountDownLatch countDownLatch = new CountDownLatch(clientTotal);        for (int i = 0; i < clientTotal; i++) {            final int count = i;            executorService.execute(()->{                try {                    semaphore.acquire();                    update(count);                    semaphore.release();                } catch (Exception e) {                    log.error("exception",e);                }                countDownLatch.countDown();            });        }        countDownLatch.await();        executorService.shutdown();        log.info("size:{}",list.size());    }    private static void update(int i){        list.add(i);    }}

上面是客户端请求 5000 次,有 200 个线程在同时请求,我使用的是 ArrayList 实现,咱们看下打印结果:

Java的CopyOnWrite怎么实现

如果是线程安全的话,那么最后的结果应该是 5000 才对,多运行几次你会发现,每次程序的执行结果都是不一样的

如果是 CopyOnWriteArrayList 呢?

@Slf4jpublic class CopyOnWriteArrayListExample {    // 请求总数    public static int clientTotal = 5000;    // 同时并发执行的线程数    public static int threadTotal = 200;    private static List<Integer> list = new CopyOnWriteArrayList<>();    public static void  main(String[] args) throws Exception{        ExecutorService executorService = Executors.newCachedThreadPool();        final Semaphore semaphore = new Semaphore(threadTotal);        final CountDownLatch countDownLatch = new CountDownLatch(clientTotal);        for (int i = 0; i < clientTotal; i++) {            final int count = i;            executorService.execute(()->{                try {                    semaphore.acquire();                    update(count);                    semaphore.release();                } catch (Exception e) {                    log.error("excepiton",e);                }                countDownLatch.countDown();            });        }        countDownLatch.await();        executorService.shutdown();        log.info("size:{}",list.size());    }    private static void update(int i){        list.add(i);    }}

关于“Java的CopyOnWrite怎么实现”这篇文章的内容就介绍到这里,感谢各位的阅读!相信大家对“Java的CopyOnWrite怎么实现”知识都有一定的了解,大家如果还想学习更多知识,欢迎关注编程网行业资讯频道。

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

Java的CopyOnWrite怎么实现

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

Java的CopyOnWrite怎么实现

这篇文章主要介绍了Java的CopyOnWrite怎么实现的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇Java的CopyOnWrite怎么实现文章都会有所收获,下面我们一起来看看吧。概念CopyOnWrite
2023-06-27

Java的CopyOnWrite集合有什么用

这篇文章主要介绍“Java的CopyOnWrite集合有什么用”,在日常操作中,相信很多人在Java的CopyOnWrite集合有什么用问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Java的CopyOnWr
2023-06-03

java怎么实现类的实例化

在Java中,类的实例化可以通过以下两种方式来完成:1. 使用`new`关键字:```javaClassName obj = new ClassName();```在这种方式下,`new`关键字用于创建类的一个新的实例。`ClassName
2023-09-21

Java怎么实现enum

在Java中,可以通过关键字`enum`来定义枚举类型。以下是实现enum的示例代码:```javapublic enum DayOfWeek {MONDAY, TUESDAY, WEDNESDAY, THURSDAY, FRIDAY, S
2023-08-21

Java的Model2应用怎么实现

今天小编给大家分享一下Java的Model2应用怎么实现的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。 几乎所有基于Java
2023-06-03

Java的Collection接口怎么实现

这篇文章主要介绍“Java的Collection接口怎么实现”,在日常操作中,相信很多人在Java的Collection接口怎么实现问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Java的Collection
2023-06-02

Java中的接口怎么实现

这篇文章主要介绍“Java中的接口怎么实现”,在日常操作中,相信很多人在Java中的接口怎么实现问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Java中的接口怎么实现”的疑惑有所帮助!接下来,请跟着小编一起来
2023-06-03

java怎么实现类的继承

在Java中,使用关键字`extends`来实现类的继承。继承允许一个类继承另一个类的属性和方法。下面是一个示例代码,演示了如何实现类的继承:```java// 定义一个父类class Animal {String name;public
2023-08-08

java的Balking模式怎么实现

本文小编为大家详细介绍“java的Balking模式怎么实现”,内容详细,步骤清晰,细节处理妥当,希望这篇“java的Balking模式怎么实现”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。Balk有拒绝,阻碍的
2023-06-29

java热加载怎么实现的

Java的热加载可以通过使用Java的反射机制和自定义的类加载器来实现。下面是一种常见的实现方式:创建一个自定义的类加载器,继承自Java的ClassLoader类,并重写findClass()方法。在findClass()方法中,根据类名
2023-10-25

java怎么实现简单的mock

在Java中,可以使用Mockito库来实现简单的mock。Mockito是一个流行的Java mocking框架,可以用于模拟对象、方法和行为。以下是一个简单的例子,演示如何使用Mockito来mock一个接口:首先,确保你的项目中已
2023-10-22

java中ScheduledExecutorService怎么实现

ScheduledExecutorService 是 JDK 提供的一个接口,用于定时执行任务。可以通过以下步骤来实现 ScheduledExecutorService:1. 首先,创建一个 ScheduledExecutorService
2023-09-14

java怎么实现延时

java实现延时的方法:1、使用Timer类Timer类负责计划任务的功能,也即指定的时间开始执行某个任务。Timer类的作用只是用于设置计划任务。Timer类的schedule方法可以按照时间计划执行程序。 public static void main(S
java怎么实现延时
2018-07-15

java怎么实现抢票

要实现抢票功能,可以采用以下步骤:1. 创建一个票池数据结构,用于存储票的信息,包括票的名称、票的数量、票的状态等。2. 创建一个线程池,用于控制并发抢票的线程数量。3. 创建抢票的线程类,实现Runnable接口或继承Thread类,在r
2023-08-31

Java列表怎么实现

这篇文章主要介绍“Java列表怎么实现”,在日常操作中,相信很多人在Java列表怎么实现问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Java列表怎么实现”的疑惑有所帮助!接下来,请跟着小编一起来学习吧!El
2023-06-03

Java多态性是怎么实现的

本篇内容主要讲解“Java多态性是怎么实现的”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Java多态性是怎么实现的”吧!多态性是通过:1 接口和实现接口并覆盖接口中同一方法的几不同的类体现的2
2023-06-17

Java的组合模式怎么实现

本篇内容介绍了“Java的组合模式怎么实现”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!1、什么是组合模式Compose objects i
2023-07-05

java怎么实现图片的展示

在Java中,可以使用JavaFX库来实现图片的展示。以下是一个简单的示例代码,演示了如何在JavaFX中展示图片:```javaimport javafx.application.Application;import javafx.sce
2023-08-20

java的工厂模式怎么实现

在Java中,工厂模式可以通过两种方式来实现:简单工厂模式和抽象工厂模式。简单工厂模式:简单工厂模式的核心是定义一个工厂类,该类负责创建其他类的实例。具体步骤如下:创建一个抽象产品类,该类定义了产品的共同属性和方法。创建具体产品类,继
java的工厂模式怎么实现
2024-02-29

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录