这些Java并发容器,你都了解吗?

文章目录

      • 前言
      • 并发容器
        • 1.ConcurrentHashMap 并发版 HashMap
          • 示例
        • 2.CopyOnWriteArrayList 并发版 ArrayList
          • 示例
        • 3.CopyOnWriteArraySet 并发 Set
          • 示例
        • 4.ConcurrentLinkedQueue 并发队列 (基于链表)
          • 示例
        • 5.ConcurrentLinkedDeque 并发队列 (基于双向链表)
          • 示例
        • 6.ConcurrentSkipListMap 基于跳表的并发 Map
          • 示例
        • 7.ConcurrentSkipListSet 基于跳表的并发 Set
          • 示例
        • 8.ArrayBlockingQueue 阻塞队列 (基于数组)
          • 示例
        • 9.LinkedBlockingQueue 阻塞队列 (基于链表)
          • 示例
        • 10.LinkedBlockingDeque 阻塞队列 (基于双向链表)
          • 示例
        • 11.PriorityBlockingQueue 线程安全的优先队列
          • 示例
        • 12.SynchronousQueue 数据同步交换的队列
          • 示例
        • 13.LinkedTransferQueue 基于链表的数据交换队列
          • 示例
        • 14.DelayQueue 延时队列
          • 示例
      • 总结
      • 写在最后

579a429daf314744b995f37351b46548

前言

在多线程环境下,数据的并发访问和修改是无法避免的问题。

为了解决这个问题,Java 提供了一系列并发容器,这些容器在内部已经处理了并发问题,使得我们可以在多线程环境下安全地访问和修改数据。


并发容器

1.ConcurrentHashMap 并发版 HashMap

最常见的并发容器之一,可以用作并发场景下的缓存。底层依然是哈希表,但在 JAVA 8 中有了不小的改变,而 JAVA 7 和 JAVA 8 都是用的比较多的版本,因此经常会将这两个版本的实现方式做一些比较(比如面试中)。

一个比较大的差异就是,JAVA 7 中采用分段锁来减少锁的竞争,JAVA 8 中放弃了分段锁,采用 CAS(一种乐观锁),同时为了防止哈希冲突严重时退化成链表(冲突时会在该位置生成一个链表,哈希值相同的对象就链在一起),会在链表长度达到阈值(8)后转换成红黑树(比起链表,树的查询效率更稳定)。

示例
import java.util.concurrent.*;public class ConcurrentHashMapExample {public static void main(String[] args) {// Creating a ConcurrentHashMapConcurrentHashMap<String, String> map = new ConcurrentHashMap<String, String>();// Adding elements to the ConcurrentHashMapmap.put("Key1", "Value1");map.put("Key2", "Value2");map.put("Key3", "Value3");// Printing the ConcurrentHashMapSystem.out.println("ConcurrentHashMap: " + map);}
}
2.CopyOnWriteArrayList 并发版 ArrayList

并发版 ArrayList,底层结构也是数组,和 ArrayList 不同之处在于:当新增和删除元素时会创建一个新的数组,在新的数组中增加或者排除指定对象,最后用新增数组替换原来的数组。

CopyOnWriteArrayList 的主要特性是,每当列表修改时,例如添加或删除元素,它都会创建列表的一个新副本。原始列表和新副本都可以进行并发读取,这样就可以在不锁定整个列表的情况下进行并发读取。这种方法在读取操作远多于写入操作的场景中非常有用。

适用场景:由于读操作不加锁,写(增、删、改)操作加锁,因此适用于读多写少的场景。

局限:由于读的时候不会加锁(读的效率高,就和普通 ArrayList 一样),读取的当前副本,因此可能读取到脏数据。如果介意,建议不用。

看看源码感受下:

image-20201023223825079

示例
import java.util.concurrent.*;public class CopyOnWriteArrayListExample {public static void main(String[] args) {// 创建一个 CopyOnWriteArrayListCopyOnWriteArrayList<String> list = new CopyOnWriteArrayList<String>();// 向 CopyOnWriteArrayList 添加元素list.add("Element1");list.add("Element2");list.add("Element3");// 打印 CopyOnWriteArrayListSystem.out.println("CopyOnWriteArrayList: " + list);}
}
3.CopyOnWriteArraySet 并发 Set

基于 CopyOnWriteArrayList 实现(内含一个 CopyOnWriteArrayList 成员变量),也就是说底层是一个数组,意味着每次 add 都要遍历整个集合才能知道是否存在,不存在时需要插入(加锁)。

CopyOnWriteArraySet 的工作原理与 CopyOnWriteArrayList 类似。每当发生修改操作(如添加或删除元素)时,它都会创建集合的一个新副本。原始集合和新副本都可以进行并发读取,这样就可以在不锁定整个集合的情况下进行并发读取。这种方法在读取操作远多于写入操作的场景中非常有用。

适用场景:在 CopyOnWriteArrayList 适用场景下加一个,集合别太大(全部遍历伤不起)。

示例
import java.util.concurrent.*;public class CopyOnWriteArraySetExample {public static void main(String[] args) {// 创建一个 CopyOnWriteArraySetCopyOnWriteArraySet<String> set = new CopyOnWriteArraySet<String>();// 向 CopyOnWriteArraySet 添加元素set.add("Element1");set.add("Element2");set.add("Element3");// 打印 CopyOnWriteArraySetSystem.out.println("CopyOnWriteArraySet: " + set);}
}
4.ConcurrentLinkedQueue 并发队列 (基于链表)

基于链表实现的并发队列,使用乐观锁 (CAS) 保证线程安全。因为数据结构是链表,所以理论上是没有队列大小限制的,也就是说添加数据一定能成功。

ConcurrentLinkedQueue 是 Java 并发包的一部分,它是基于链接节点的无界线程安全队列。它按照 FIFO(先进先出)的原则对元素进行排序。

ConcurrentLinkedQueue 的主要优点是它允许完全并发的插入,并且使用了一种高效的“wait-free”算法。

示例
import java.util.concurrent.*;public class ConcurrentLinkedQueueExample {public static void main(String[] args) {// 创建一个 ConcurrentLinkedQueueConcurrentLinkedQueue<String> queue = new ConcurrentLinkedQueue<String>();// 向 ConcurrentLinkedQueue 添加元素queue.add("Element1");queue.add("Element2");queue.add("Element3");// 打印 ConcurrentLinkedQueueSystem.out.println("ConcurrentLinkedQueue: " + queue);}
}
5.ConcurrentLinkedDeque 并发队列 (基于双向链表)

基于双向链表实现的并发队列,可以分别对头尾进行操作,因此除了先进先出 (FIFO),也可以先进后出(FILO),当然先进后出的话应该叫它栈了。

ConcurrentLinkedDeque 是 Java 并发包的一部分,它是一个基于链接节点的无界并发双端队列。在 ConcurrentLinkedDeque 中,添加、删除等操作可以在队列的两端进行,使其具有更高的并发性。

示例
import java.util.concurrent.*;public class ConcurrentLinkedDequeExample {public static void main(String[] args) {// 创建一个 ConcurrentLinkedDequeConcurrentLinkedDeque<String> deque = new ConcurrentLinkedDeque<String>();// 向 ConcurrentLinkedDeque 添加元素deque.add("Element1");deque.addFirst("Element2");deque.addLast("Element3");// 打印 ConcurrentLinkedDequeSystem.out.println("ConcurrentLinkedDeque: " + deque);}
}
6.ConcurrentSkipListMap 基于跳表的并发 Map

ConcurrentSkipListMap 是 Java 并发包的一部分,它是一个线程安全的排序映射表。它使用跳表的数据结构来保证元素的有序性和并发性。

跳表是一种可以进行二分查找的有序链表。ConcurrentSkipListMap 提供了预期的平均 log(n) 时间成本来执行 containsKeygetputremove 操作,并且它的并发性通常优于基于树的算法。

SkipList 即跳表,跳表是一种空间换时间的数据结构,通过冗余数据,将链表一层一层索引,达到类似二分查找的效果

image-20201023223853743

示例
import java.util.concurrent.*;public class ConcurrentSkipListMapExample {public static void main(String[] args) {// 创建一个 ConcurrentSkipListMapConcurrentSkipListMap<String, String> map = new ConcurrentSkipListMap<String, String>();// 向 ConcurrentSkipListMap 添加元素map.put("Key1", "Value1");map.put("Key2", "Value2");map.put("Key3", "Value3");// 打印 ConcurrentSkipListMapSystem.out.println("ConcurrentSkipListMap: " + map);}
}
7.ConcurrentSkipListSet 基于跳表的并发 Set

类似 HashSet 和 HashMap 的关系,ConcurrentSkipListSet 里面就是一个 ConcurrentSkipListMap,

ConcurrentSkipListSet 是 Java 并发包的一部分,它是一个线程安全的排序集合。它使用跳表的数据结构来保证元素的有序性和并发性。

跳表是一种可以进行二分查找的有序链表。ConcurrentSkipListSet 提供了预期的平均 log(n) 时间成本来执行 containsaddremove 操作,并且它的并发性通常优于基于树的算法。

示例
import java.util.concurrent.*;public class ConcurrentSkipListSetExample {public static void main(String[] args) {// 创建一个 ConcurrentSkipListSetConcurrentSkipListSet<String> set = new ConcurrentSkipListSet<String>();// 向 ConcurrentSkipListSet 添加元素set.add("Element1");set.add("Element2");set.add("Element3");// 打印 ConcurrentSkipListSetSystem.out.println("ConcurrentSkipListSet: " + set);}
}
8.ArrayBlockingQueue 阻塞队列 (基于数组)

ArrayBlockingQueue 是 Java 并发包的一部分,它是一个基于数组的有界阻塞队列。此队列按 FIFO(先进先出)原则对元素进行排序。

ArrayBlockingQueue 在尝试插入元素到已满队列或从空队列中移除元素时,会导致线程阻塞,直到有空间或元素可用。

基于数组实现的可阻塞队列,构造时必须制定数组大小,往里面放东西时如果数组满了便会阻塞直到有位置(也支持直接返回和超时等待),通过一个锁 ReentrantLock 保证线程安全。

image-20201023223912400

乍一看会有点疑惑,读和写都是同一个锁,那要是空的时候正好一个读线程来了不会一直阻塞吗?

答案就在 notEmpty、notFull 里,这两个出自 lock 的小东西让锁有了类似 synchronized + wait + notify 的功能。传送门 → 终于搞懂了 sleep/wait/notify/notifyAll

示例
import java.util.concurrent.*;public class ArrayBlockingQueueExample {public static void main(String[] args) {// 创建一个 ArrayBlockingQueueArrayBlockingQueue<String> queue = new ArrayBlockingQueue<String>(3);// 向 ArrayBlockingQueue 添加元素try {queue.put("Element1");queue.put("Element2");queue.put("Element3");} catch (InterruptedException e) {e.printStackTrace();}// 打印 ArrayBlockingQueueSystem.out.println("ArrayBlockingQueue: " + queue);}
}
9.LinkedBlockingQueue 阻塞队列 (基于链表)

LinkedBlockingQueue 是 Java 并发包的一部分,它是一个基于链表的可选有界阻塞队列。此队列按照 FIFO(先进先出)的原则对元素进行排序。

LinkedBlockingQueue 在尝试插入元素到已满队列或从空队列中移除元素时,会导致线程阻塞,直到有空间或元素可用。

基于链表实现的阻塞队列,想比与不阻塞的 ConcurrentLinkedQueue,它多了一个容量限制,如果不设置默认为 int 最大值。

示例
import java.util.concurrent.*;public class LinkedBlockingQueueExample {public static void main(String[] args) {// 创建一个 LinkedBlockingQueueLinkedBlockingQueue<String> queue = new LinkedBlockingQueue<String>(3);// 向 LinkedBlockingQueue 添加元素try {queue.put("Element1");queue.put("Element2");queue.put("Element3");} catch (InterruptedException e) {e.printStackTrace();}// 打印 LinkedBlockingQueueSystem.out.println("LinkedBlockingQueue: " + queue);}
}
10.LinkedBlockingDeque 阻塞队列 (基于双向链表)

LinkedBlockingDeque 是 Java 并发包的一部分,它是一个基于链表的可选有界阻塞双端队列。此队列按照 FIFO(先进先出)的原则对元素进行排序。

LinkedBlockingDeque 在尝试插入元素到已满队列或从空队列中移除元素时,会导致线程阻塞,直到有空间或元素可用。双端队列的优势在于可以从两端插入或移除元素。

类似 LinkedBlockingQueue,但提供了双向链表特有的操作。

示例
import java.util.concurrent.*;public class LinkedBlockingDequeExample {public static void main(String[] args) {// 创建一个 LinkedBlockingDequeLinkedBlockingDeque<String> deque = new LinkedBlockingDeque<String>(3);// 向 LinkedBlockingDeque 添加元素try {deque.putFirst("Element1");deque.putLast("Element2");deque.putFirst("Element3");} catch (InterruptedException e) {e.printStackTrace();}// 打印 LinkedBlockingDequeSystem.out.println("LinkedBlockingDeque: " + deque);}
}
11.PriorityBlockingQueue 线程安全的优先队列

PriorityBlockingQueue 是 Java 并发包的一部分,它是一个无界的并发队列。它使用了和类 java.util.PriorityQueue 一样的排序规则,并且能够确保在并发环境下的线程安全。

PriorityBlockingQueue 中的元素按照自然顺序或者由比较器提供的顺序进行排序。队列不允许使用 null 元素。

构造时可以传入一个比较器,可以看做放进去的元素会被排序,然后读取的时候按顺序消费。某些低优先级的元素可能长期无法被消费,因为不断有更高优先级的元素进来。

示例
import java.util.concurrent.*;public class PriorityBlockingQueueExample {public static void main(String[] args) {// 创建一个 PriorityBlockingQueuePriorityBlockingQueue<String> queue = new PriorityBlockingQueue<String>();// 向 PriorityBlockingQueue 添加元素queue.add("Element1");queue.add("Element2");queue.add("Element3");// 打印 PriorityBlockingQueueSystem.out.println("PriorityBlockingQueue: " + queue);}
}
12.SynchronousQueue 数据同步交换的队列

SynchronousQueue 是 Java 并发包的一部分,它是一个不存储元素的阻塞队列。每一个 put 操作必须等待一个 take 操作,否则不能继续添加元素,反之亦然。

这种特性使 SynchronousQueue 成为线程之间传递数据的好工具。它可以看作是一个传球手,负责把生产者线程处理的数据直接传递给消费者线程。

一个虚假的队列,因为它实际上没有真正用于存储元素的空间,每个插入操作都必须有对应的取出操作,没取出时无法继续放入。

示例
import java.util.concurrent.SynchronousQueue;public class Main {public static void main(String[] args) {SynchronousQueue<Integer> queue = new SynchronousQueue<>();new Thread(()->{try{for(int i=0;;i++){System.out.println("放入:" + i);queue.put(i);}}catch (InterruptedException e){e.printStackTrace();}}).start();new Thread(()->{try{while(true){System.out.println("取出:" + queue.take());Thread.sleep((long)(Math.random()*2000));}}catch (InterruptedException e){e.printStackTrace();}}).start();}
}

运行结果:

取出:0
放入:0
取出:1
放入:1
放入:2
取出:2
取出:3
放入:3
取出:4
放入:4
...
...

可以看到,写入的线程没有任何 sleep,可以说是全力往队列放东西,而读取的线程又很不积极,读一个又 sleep 一会。输出的结果却是读写操作成对出现。

JAVA 中一个使用场景就是 Executors.newCachedThreadPool(),创建一个缓存线程池。

image-20201023223932760

13.LinkedTransferQueue 基于链表的数据交换队列

LinkedTransferQueue 是 Java 并发包的一部分,它是一个由链表结构组成的无界转移阻塞队列。队列按照 FIFO(先进先出)的原则对元素进行排序。

LinkedTransferQueue 的一个特性是,它可以尝试将元素直接转移给消费者,如果没有等待的消费者,元素就会被添加到队列的尾部,等待消费者来获取。

实现了接口 TransferQueue,通过 transfer 方法放入元素时,如果发现有线程在阻塞在取元素,会直接把这个元素给等待线程。如果没有人等着消费,那么会把这个元素放到队列尾部,并且此方法阻塞直到有人读取这个元素。和 SynchronousQueue 有点像,但比它更强大。

示例
import java.util.concurrent.*;public class LinkedTransferQueueExample {public static void main(String[] args) {// 创建一个 LinkedTransferQueueLinkedTransferQueue<String> queue = new LinkedTransferQueue<String>();// 启动一个新线程来从 LinkedTransferQueue 取出元素new Thread(() -> {try {System.out.println("Taken: " + queue.take());} catch (InterruptedException e) {e.printStackTrace();}}).start();// 向 LinkedTransferQueue 添加一个元素try {queue.transfer("Element");} catch (InterruptedException e) {e.printStackTrace();}}
}
14.DelayQueue 延时队列

DelayQueue 是 Java 并发包的一部分,它是一个无界阻塞队列,只有在延迟期满时才能从中提取元素。此队列的头部是延迟期满后保存时间最长的元素。如果延迟都还没有期满,则队列没有头部,并且 poll 将返回 null

元素在 DelayQueue 中的顺序是按照其到期时间的先后顺序进行排序的,越早到期的元素越排在队列前面。延迟队列常用于实现定时任务功能。

可以使放入队列的元素在指定的延时后才被消费者取出,元素需要实现 Delayed 接口。

示例
import java.util.concurrent.*;public class DelayQueueExample {public static void main(String[] args) {// 创建一个 DelayQueueDelayQueue<DelayedElement> queue = new DelayQueue<DelayedElement>();// 向 DelayQueue 添加一个元素,延迟 3 秒queue.put(new DelayedElement(3000, "Element"));// 从 DelayQueue 获取元素try {DelayedElement element = queue.take();System.out.println("Taken: " + element);} catch (InterruptedException e) {e.printStackTrace();}}
}class DelayedElement implements Delayed {private long delayTime; // 延迟时间private long expire;  // 到期时间private String element; // 元素数据public DelayedElement(long delay, String element) {this.delayTime = delay;this.element = element;this.expire = System.currentTimeMillis() + delay;}@Overridepublic long getDelay(TimeUnit unit) {return unit.convert(expire - System.currentTimeMillis(), TimeUnit.MILLISECONDS);}@Overridepublic int compareTo(Delayed o) {return (int) (this.getDelay(TimeUnit.MILLISECONDS) - o.getDelay(TimeUnit.MILLISECONDS));}@Overridepublic String toString() {return element;}
}

总结

从上面的介绍总总结有以下几种容器类

  1. ConcurrentHashMap:并发版 HashMap

  2. CopyOnWriteArrayList:并发版 ArrayList

  3. CopyOnWriteArraySet:并发 Set

  4. ConcurrentLinkedQueue:并发队列 (基于链表)

  5. ConcurrentLinkedDeque:并发队列 (基于双向链表)

  6. ConcurrentSkipListMap:基于跳表的并发 Map

  7. ConcurrentSkipListSet:基于跳表的并发 Set

  8. ArrayBlockingQueue:阻塞队列 (基于数组)

  9. LinkedBlockingQueue:阻塞队列 (基于链表)

  10. LinkedBlockingDeque:阻塞队列 (基于双向链表)

  11. PriorityBlockingQueue:线程安全的优先队列

  12. SynchronousQueue:读写成对的队列

  13. LinkedTransferQueue:基于链表的数据交换队列

  14. DelayQueue:延时队列

Java 并发容器为处理多线程环境下的数据访问和修改提供了强大的工具。

通过了解和学习这些并发容器,我们可以更好地理解并发编程,更有效地处理并发问题。

无论你是正在学习 Java,还是已经在使用 Java 进行开发,我都强烈建议你深入了解这些并发容器,它们将在你的并发编程之路上起到重要的作用。


写在最后

感谢您的支持和鼓励! 😊🙏

如果大家对相关文章感兴趣,可以关注公众号"架构殿堂",会持续更新AIGC,java基础面试题, netty, spring boot, spring cloud等系列文章,一系列干货随时送达!

csdn-end

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/203704.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Vue学习计划-Vue2--Vue核心(五)条件、列表渲染、表单数据

1. 条件渲染 v-if v-if“表达式”v-else-if “表达式”v-else “表达式” 适用于&#xff1a;切换频率较低的场景 特点&#xff1a;不显示dom元素&#xff0c;直接被删除 注意&#xff1a;v-if和v-else-if、v-else一起使用&#xff0c;但要求结构不能被打断 v-if和template一…

Android笔记(十七):PendingIntent简介

PendingIntent翻译成中文为“待定意图”&#xff0c;这个翻译很好地表示了它的涵义。PendingIntent描述了封装Intent意图以及该意图要执行的目标操作。PendingIntent封装Intent的目标行为的执行是必须满足一定条件&#xff0c;只有条件满足&#xff0c;才会触发意图的目标操作。…

Kotlin 中的 also 和 run:选择正确的作用域函数

在 Kotlin 中&#xff0c;also 和 run 是两个十分有用的作用域函数。 虽然它们在功能上相似&#xff0c;但各自有独特的用途和适用场景。 一、分析&#xff1a; also&#xff1a;在对象的上下文中执行给定的代码块&#xff0c;并返回对象本身。它的参数是一个接收对象并返回…

分布式分布式事务分布式锁分布式ID

目录 分布式分布式系统设计理念目标设计思路中心化去中心化 基本概念分布式与集群NginxRPC消息中间件&#xff08;MQ&#xff09;NoSQL&#xff08;非关系型数据库&#xff09; 分布式事务1 事务2 本地事务3 分布式事务4 本地事务VS分布式事务5 分布式事务场景6 CAP原理7 CAP组…

ChatGPT发展历程

ChatGPT是一个在2020年成立的在线聊天平台&#xff0c;它的发展历程如下&#xff1a; 初期阶段&#xff1a;2020年&#xff0c;在全球疫情爆发的情况下&#xff0c;ChatGPT创始人开始思考如何为人们提供一个快捷、安全、便利的在线聊天平台。他们选择使用GPT&#xff08;生成对…

(2/2)敏捷实践指南 Agile Practice Guide ([美] Project Management institute 著)

附录 A1 - 《PMBOK指南》映射 表A1显示了第六版《PMBOK指南》中定义的项目管理过程组与知识领域之间的对应关系 本附录说明了如何利用混合和敏捷方法处理《PMBOK指南》知识领域&#xff08;请参见表A1-2&#xff09;中所述的属性&#xff0c;其中涵盖了相同和不同的属性&…

conda 安装教程分享

大家好&#xff0c;我是微赚淘客系统的小编&#xff0c;也是冬天不穿秋裤&#xff0c;天冷也要风度的程序猿&#xff01;今天我将向大家介绍如何使用conda进行安装。 首先&#xff0c;让我们来了解一下conda。conda是Anaconda发行版的一部分&#xff0c;它是一个开源的包管理系…

为什么那些删库跑路的人都极其下流

为什么那些删库跑路的人都极其下流&#xff1f;因为真的下流。注意&#xff0c;我不是针对跑路者的人品&#xff0c;遇到不公正待遇不敢反抗本身就比下流还下流&#xff0c;我说的是这种对抗方式太多低等。 干不好是能力问题&#xff0c;你不干是态度问题。记住我的话&#xf…

golang版aes-cbc-pkcs7加密解密base64hex字符串输入输出

最近项目中golang项目中使用aes加解密&#xff0c;做个记录方便以后使用 aes-cbc-pkcs7加密解密base64输入输出 type AesBase64 struct {key []byte // 允许16,24,32字节长度iv []byte // 只允许16字节长度 }func NewAesBase64(key []byte, iv []byte) *AesBase64 {return &…

C#网络应用程序(Web页面浏览器、局域网聊天程序)

目录 一、创建Web页面浏览器 1.示例源码 2.生成效果 二、局域网聊天程序 1.类 2.服务器端 3.客户端 一、创建Web页面浏览器 TextBox 控件用来输入要浏览的网页地址&#xff0c;Button控件用来执行浏览网页操作&#xff0c; WebBrowser控件用来显示要浏览的网页。这个控…

Matlab 曲线动态绘制

axes(handles.axes1); % 选定所画坐标轴 figure也可 h1 animatedline; h1.Color b; h1.LineWidth 2; h1.LineStyle -; % 线属性设置 for i 1 : length(x)addpoints(h1,x(i),y(i)); % x/y为待绘制曲线数据drawnow;pause(0.01); % 画点间停顿 end 示例&#xff1a; figure…

exynos4412—中断处理

一、什么是中断 一种硬件上的通知机制&#xff0c;用来通知CPU发生了某种需要立即处理的事件 分为&#xff1a; 内部中断 CPU执行程序的过程中&#xff0c;发生的一些硬件出错、运算出错事件&#xff08;如分母为0、溢出等等&#xff09;&#xff0c;不可屏蔽外部中断 外设发…

scitb包1.6版本发布,一个为制作专业统计表格而生的R包

目前&#xff0c;本人写的scitb包1.6版本已经正式在R语言官方CRAN上线&#xff0c;scitb包是一个为生成专业化统计表格而生的R包。 可以使用以下代码安装 install.packages("scitb")安装过旧版本的从新安装一次就可以升级了,根据粉丝的建议&#xff0c;增加了Overal…

RocketMQ-RocketMQ集群实践(搭建)

搭建RocketMQ可视化管理服务 下载可视化客户端源码下载 | RocketMQ 这里只提供了源码&#xff0c;并没有提供直接运行的jar包。将源码下载下来后&#xff0c;需要解压并进入对应的目录&#xff0c;使用maven进行编译。(需要提前安装maven客户端) mvn clean package -Dmaven.t…

手动部署1个Cloud Run service

什么是Cloud Run 来自chatgpt&#xff1a; Google Cloud Run 是一项全托管的服务器托管平台&#xff0c;它允许您在容器化的环境中运行无服务器应用程序。Cloud Run 提供了一种简单而灵活的方式来构建、部署和扩展应用程序&#xff0c;无需管理底层基础设施。 以下是 Cloud …

sql常用语法练习

表名word namecontinentareapopulationgdpAfghanistanAsia6522302550010020343000000AlbaniaEurope28748283174112960000000AlgeriaAfrica238174137100000188681000000AndorraEurope468781153712000000AngolaAfrica124670020609294100990000000.... 基础练习 基础查询 1、阅…

RocketMq实战(待完善)

目录 生产者 发送消息固定步骤 发送模式 1. 单向发送 2. 同步发送 3. 异步发送 生产消息完整代码 消费者 消费消息固定步骤 简单消费代码示例 消息模型 广播消息 顺序消息 延迟消息 批量消息 事务消息 生产者 发送消息固定步骤 1.创建消息生产者producer&#…

操作系统的运行机制+中断和异常

一、CPU状态 在CPU设计和生产的时候就划分了特权指令和非特叔指令&#xff0c;因此CPU执行一条指令前就能断出其类型 CPU有两种状态&#xff0c;“内核态”和“用户态” 处于内核态时&#xff0c;说明此时正在运行的是内核程序&#xff0c;此时可以执行特权指令。 处于用户态…

Jenkins+Maven+Gitlab+Tomcat 自动化构建打包,部署

环境准备Jenkins工具、环境、插件配置全局变量配置安装插件Deploy to containerMaven Integration plugin配置国内mvn源 创建maven项目 环境准备 1、安装服务 Jenkins工具、环境、插件配置 全局变量配置 Manage Jenkins>tools>JDK 安装 安装插件 Deploy to contai…

231206日课:高原反应的第三天

早冥读写跑&#xff08;早起、冥想、阅读、写作、跑步&#xff09; 践行第三天 一、知识基础 早起 冥想 阅读 写作 运动 二、个人化运用 日程复盘 优秀的AI生成的表格 汇总信息时间段安排睡眠时间20:54-04:57睡眠时间 (8小时3分钟)早晨时间安排04:57-05:00起床、洗漱早晨时间安…