JUC 是学习 Java 并发编程的小伙伴不可避免的一个 pkg,JUC提供了对并发编程的底层支持,比如我们熟悉的线程池、MQ、线程同步... 都有JUC的影子,下面我们一起来看看JUC下比较重要的几个class。

CountdownLatch

先看一下 **latch **是什么意思:



门闩大家都知道吧,假设我们现在有一扇铁门,每个线程跑到这铁门时都取下一把门闩,当所有的门闩都从门上取下来时,门就打开了,线程就可以继续往下执行。

public class CountdownLatchTest {
public static void main(String[] args) throws InterruptedException {
// 传入放多少个门闩
CountDownLatch cdl = new CountDownLatch(10); // 启 10 个线程
for (int i = 0; i < 10; i++) {
new Thread(() -> {
System.out.println(Thread.currentThread().getName() + " 取下一把门闩");
// 每个线程运行完就取下一把门闩
cdl.countDown();
}, "线程" + i).start();
}
// 铁门在这等着门闩被取完了才打开
cdl.await(); System.out.println("门打开了");
}
}

运行结果:



在实际开发中,我们常用多个门闩来对多个线程的执行顺序进行控制,后面有机会给大家分享一下案例。

CyclicBarrier

我们要搞清楚一个东西肯定得知道他是什么意思,大佬对命名都是很讲究的。(我大学刚接触编程都是 a,b,c,d,e,f,g命名)



单词每个都认识,连在一起就不知道是什么意思了。

循环的障碍?循环的屏障?其实都无所谓,自己挑一个容易理解的名字即可,我理解的是“循环的栅栏”

什么是栅栏?

就是拦在你前面的一个障碍物,这个障碍物你一个人推不开,得多叫几个兄弟一起来推才行,推开了才能继续往前走。

public class CyclicBarrierTest {
public static void main(String[] args) {
CyclicBarrier cb = new CyclicBarrier(10, () -> {
// todo:人齐了该干嘛
System.out.println("人齐了,开冲");
}); for (int i = 0; i < 10; i++) {
new Thread(() -> {
System.out.println(Thread.currentThread().getName() + "已就位");
try {
// 线程都在这里等着,直到人气才会执行 CyclicBarrier 里面的方法
cb.await();
} catch (InterruptedException e) {
e.printStackTrace();
} catch (BrokenBarrierException e) {
e.printStackTrace();
}
}, i + " 号兄弟").start();
}
}
}

运行结果:



CountDownLatch 不同的是,CyclicBarrier 还有个 Cyclic ,我相信你已经猜到了,我们启动 20 个线程看看



门闩是一次性的,当门闩全部取下来这个门就打开了,而栅栏是有弹性的,当人满了把它推开后,它又弹回来把路又堵住了,又需要 n 个人才能把栅栏推开。

如果有小伙伴感兴趣可以去看下它是怎么实现循环的

Semaphore

Semaphore 的意思是:信号量,信号量在线程同步中也是很常见的,你可以把它看作是一个 计数器 cnt

  • 当 cnt > 0 说明还有资源
  • 当 cnt < 0 说明没有资源了

举个栗子:比如办公室有一台打印机,但是今天领导让大家写一份加薪报告,大家都忙着用这台打印机打印自己的报告,准备今晚加餐吃鸡腿。

public class SemaphoreTest {
public static void main(String[] args) {
// 只有一台打印机
Semaphore s = new Semaphore(1); for (int i = 0; i < 20; i++) {
new Thread(() -> {
try {
// 去抢打印机(不一定抢得到)
s.acquire();
System.out.println(Thread.currentThread().getName() + " 拿到打印机了,今晚可以加鸡腿了");
// 把打印机还回去
s.release();
} catch (InterruptedException e) {
e.printStackTrace();
}
}, "员工 " + i).start();
}
} }

输出结果:



**Semaphore **适用于资源有限,消费者无限的场景,比如打印机就是一个很好的例子。

因为我初始化的资源是 1 ,所以会让你觉得和 差不多,其实不然

资源还可以初始化为 10 20 50 100.... 你可能想到了什么,没错,就是限流!

有大量的请求进来,我们只允许我们指定的数量请求进行处理,更多的用途就等以后你来开发了

ReadWriteLock

读写锁

读锁:也就是共享锁,这把锁允许多个线程持有

写锁:也就是独占锁,这把锁只允许一个线程拥有

说人话的意思就是:读不加锁,写加锁

我们都知道加锁的效率会比较低,如果我们只是对资源进行读取,其实是没有线程安全的问题,这时候再加锁就显得有点脱了裤子放屁(多此一举)

所以我们就有了读写锁:读不加锁,写加锁

public class ReadWriteLockTest {
private static final ReentrantReadWriteLock readWriteLock = new ReentrantReadWriteLock();
// 读写锁通过 ReentrantReadWriteLock 实例进行创建,说明读写锁也是可重入的
private static final ReentrantReadWriteLock.ReadLock readLock = readWriteLock.readLock();
private static final ReentrantReadWriteLock.WriteLock writeLock = readWriteLock.writeLock(); public static void main(String[] args) {
ReadWriteLockTest o = new ReadWriteLockTest(); // 先把 写方法注释掉,模拟读操作
for (int i = 0; i < 10; i++) {
new Thread(() -> {
read();
//write();
}, "线程" + i).start();
} } public static void read() {
// 上读锁
readLock.lock();
try {
System.out.println(Thread.currentThread().getName() + " 开始读取数据");
System.out.println(Thread.currentThread().getName() + " 读取数据成功");
} finally {
readLock.unlock();
} } public static void write() {
writeLock.lock();
try {
System.out.println(Thread.currentThread().getName() + "开始写入数据");
System.out.println(Thread.currentThread().getName() + "写入数据成功");
} finally {
writeLock.unlock();
}
}
}

读操作结果:



我们可以看到执行顺序,线程 8 还没读取成功,线程 9 又在读取数据了。

写操作结果:



我们可以看到,虽然线程不是顺序执行的,但是 新的线程 一定是 等上一个线程写入成功后,自己才开始写如数据

也就是说,开始写入和写入成功永远是成对出现。

BlockingQueue

阻塞队列:相信大家对 Java 中的容器一定不陌生,容器又分了两类:Collection && Map

Collection 中的 List && Set ** **大家都已经耳熟能详了,但是 Queue 可能很少用到

但是我可以告诉你,Queue 的重要程度丝毫不亚于你所知道另外几种

BlockingQueue 更是 线程池MQ 发展的基石

复习一下 Queue,Queue 是一个队列:从尾进,从头出(先进先出)

Queue 给我们提供了几个API:

  • add() 添加一个元素,队列满则报错
  • offer() 添加一个元素,队列满不报错
  • poll() 移除队头元素
  • peek() 获取队头元素

而BlockingQueue 给我们提供了几个阻塞的API:

  • put() 往队列里面放元素,如果队列满,就一直等着(阻塞),直到队列空出来位置。
  • take() 往队列里面取元素,如果队列为空,就一直等着(阻塞),直到队列有元素加进来

还记得线程池核心参数中的任务队列吗?就是让我们传入一个 BlockingQueue

BlockingQueue 分为有界和无界两种:

  • 有界: ArrayBlockingQueue (数组实现,要指定数组长度,当然有界啦)
  • 无界:LinkedBlockingQueue(链表实现,新任务直接挂在尾节点上,当然就无界了)

说了这么多,我们浅写一个 生产者/消费者 模型:

public class BlockingQueueTest {
public static void main(String[] args) {
// 实例化一个容量为 5 的 BlockingQueue
ArrayBlockingQueue<Integer> abq = new ArrayBlockingQueue<>(5); new Thread(() -> {
for (int i = 0; i < 20; i++) {
try {
abq.put(i);
System.out.println("生产者生产数据:" + i);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}).start(); new Thread(() -> {
for (int i = 0; i < 20; i++) {
try {
Integer take = abq.take();
System.out.println("消费者消费数据:" + take);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}).start(); }
}

运行结果:



这就是一个简单的 MQ

SynchronousQueue

SynchronousQueue 是一个很特殊的 BlockingQueue,特殊在:



它是没有容量,也就是说:生产者把生产出来的数据,手把手递给消费者才算完成

conclusion

其实还有 DelayQueue(延时队列),TransferQueue(控制生产者生产速率用) ... 没有讲,今天我累了 TAT,下节准备分享一下 线程池相关的知识,铁铁们,点个赞再走吧。

Java并发编程(一)JUC同步类的更多相关文章

  1. 【Java并发编程二】同步容器和并发容器

    一.同步容器 在Java中,同步容器包括两个部分,一个是vector和HashTable,查看vector.HashTable的实现代码,可以看到这些容器实现线程安全的方式就是将它们的状态封装起来,并 ...

  2. java并发编程:线程安全管理类--原子操作类--AtomicInteger

    在java并发编程中,会出现++,--等操作,但是这些不是原子性操作,这在线程安全上面就会出现相应的问题.因此java提供了相应类的原子性操作类. 1.AtomicInteger

  3. Java并发编程笔记之Unsafe类和LockSupport类源码分析

    一.Unsafe类的源码分析 JDK的rt.jar包中的Unsafe类提供了硬件级别的原子操作,Unsafe里面的方法都是native方法,通过使用JNI的方式来访问本地C++实现库. rt.jar ...

  4. java并发编程:线程同步和锁

    一.锁的原理 java中每个对象都有一个内置锁.当程序运行到非静态的synchronized同步方法上时,自动获得与正在执行代码类的当前实例(this)有关的锁.获得一个对象的锁也称为获取锁,当程序运 ...

  5. java并发编程基础——线程同步

    线程同步 一.线程安全问题 如果你的代码所在的进程中有多个线程在同时运行,而这些线程可能会同时运行这段代码.如果每次运行结果和单线程运行的结果是一样的,而且其他的变量的值也和预期的是一样的,就是线程安 ...

  6. Java并发编程阅读笔记-同步容器、工具类整理

  7. Java并发编程(3) JUC中的锁

    一 前言 前面已经说到JUC中的锁主要是基于AQS实现,而AQS(AQS的内部结构 .AQS的设计与实现)在前面已经简单介绍过了.今天记录下JUC包下的锁是怎么基于AQS上实现的 二 同步锁 同步锁不 ...

  8. Java并发编程,互斥同步和线程之间的协作

    互斥同步和线程之间的协作 互斥同步 Java 提供了两种锁机制来控制多个线程对共享资源的互斥访问,第一个是 JVM 实现的 synchronized,而另一个是 JDK 实现的 ReentrantLo ...

  9. java并发编程:线程安全管理类--原子包--java.util.concurrent.atomic

    java.util.concurrent.atomic 的描述 AtomicBoolean 可以用原子方式更新的 boolean 值. AtomicInteger 可以用原子方式更新的 int 值. ...

  10. java并发编程:线程安全管理类--原子操作类--AtomicBoolean

    1.类AtomicBoolean

随机推荐

  1. linux 性能自我学习 ———— 软中断 [五]

    前言 linux 性能的自我学习. 正文 什么是软中断呢? 举一个网络的例子. linux 将中断处理过程分为两个阶段: 上半部用来快速处理中断,他在中断禁止模式下运行,注意是处理跟硬件紧密相关或时间 ...

  2. 重新整理数据结构与算法(c#)—— 算法套路二分法[二十四]

    前言 前面写过二分法使用的是递归手法,然后该节才有不递归的方式执行. 二分法的时间复杂度为log(2)n.空间复杂度为:1. 正文 代码: static void Main(string[] args ...

  3. Vue3开源组件库

    最近收到的很多问题都是关于Vue3组件库的问题 今天就给大家推荐几个基于Vue3重构的开源组件库 目前状态都处于Beta阶段,建议大家抱着学习的心态入场,勿急于用到生产环境 Ant-design-vu ...

  4. 力扣744(java&python)- 寻找比目标字母大的最小字母(简单)

    题目: 给你一个排序后的字符列表 letters ,列表中只包含小写英文字母.另给出一个目标字母 target,请你寻找在这一有序列表里比目标字母大的最小字母. 在比较时,字母是依序循环出现的.举个例 ...

  5. 动态尺寸模型优化实践之Shape Constraint IR Part I

    简介: 在本系列分享中我们将介绍BladeDISC在动态shape语义下做性能优化的一些实践和思考.本次分享的是我们最近开展的有关shape constraint IR的工作,Part I 中我们将介 ...

  6. Serverless 时代下微服务应用全托管解决方案

    简介: 本文介绍了 Serverless 时代下微服务的发展以及过程中遇到的相对较复杂的需求,面对这些,阿里云 Serverless 应用引擎 SAE 将"Serverless"的 ...

  7. Quick BI V4.0功能“炸弹”来袭,重磅推出即席分析、模板市场、企业微信免密登录等强势功能

    简介: 2021年7月,Quick BI公共云版本迭代新功能:重磅推出即席分析.模板市场,分析门槛再降低:推出企业微信无缝对接,移动端类目个性配置及管理提升多端能力:数据建模配置交互升级至拖拽模式提升 ...

  8. 重磅发布|新一代云原生数据仓库AnalyticDB「SQL智能诊断」功能详解

    ​简介: AnalyticDB For MySQL为用户提供了高效.实时.功能丰富并且智能化的「SQL智能诊断」和「SQL智能调优」功能,提供用户SQL性能调优的思路.方向和具体的方法,降低用户使用成 ...

  9. Flink 在顺丰的应用实践

    ​简介: 顺丰基于 Flink 建设实时数仓的思路,引入 Hudi On Flink 加速数仓宽表,以及实时数仓平台化建设的实践. 本⽂由社区志愿者苗文婷整理,内容源⾃顺丰科技大数据平台研发工程师龙逸 ...

  10. [GPT] Unable to negotiate with xx.xx.xx.xx port 22: no matching host key type found. Their offer: ssh-rsa,ssh-dss

      这个错误通常发生在 SSH 客户端无法找到与 SSH服务器 匹配的主机密钥类型时. 这可能是因为SSH服务器配置为使用SSH客户端不支持的主机密钥类型. 要解决此问题,您需要将缺少的主机密钥类型添 ...