如何理解 Java 并发编程中的 CyclicBarrier 和 ReentrantLock?
cac55 2024-09-19 17:04 31 浏览 0 评论
在 Java 并发编程中,如何有效地控制多个线程的协作与同步是一个重要的课题。本文将详细探讨两个关键工具类:CyclicBarrier 和 ReentrantLock。通过对其原理、源码和应用场景的深入分析,希望能帮助读者更好地理解和应用这些工具类。
CyclicBarrier 的应用与原理解析
CyclicBarrier 是 Java 并发编程中的一个重要工具类,用于协调多个线程在某个同步点共同等待,直到所有线程都到达同步点后才继续执行。它的主要用途是让一组线程在某个固定点上等待,直到所有线程都到达此点后再一起继续执行,类似于“栅栏”的功能。
基本用法
CyclicBarrier 的构造函数主要有两个:
- CyclicBarrier(int parties):创建一个新的 CyclicBarrier,其计数器的初始值为 parties。
- CyclicBarrier(int parties, Runnable barrierAction):创建一个新的 CyclicBarrier,其计数器的初始值为 parties,并且在所有线程都到达屏障时执行指定的 barrierAction 任务。
java
import java.util.concurrent.BrokenBarrierException;
import java.util.concurrent.CyclicBarrier;
public class CyclicBarrierExample {
public static void main(String[] args) {
// 创建一个CyclicBarrier,指定需要同步的线程数为3
CyclicBarrier barrier = new CyclicBarrier(3, () -> {
// 所有线程到达屏障后执行的任务
System.out.println("所有线程都到达屏障,继续执行...");
});
// 创建并启动三个线程
for (int i = 0; i < 3; i++) {
new Thread(new Worker(barrier)).start();
}
}
}
class Worker implements Runnable {
private CyclicBarrier barrier;
public Worker(CyclicBarrier barrier) {
this.barrier = barrier;
}
@Override
public void run() {
try {
System.out.println(Thread.currentThread().getName() + " 正在执行");
Thread.sleep(1000); // 模拟任务执行
System.out.println(Thread.currentThread().getName() + " 到达屏障");
barrier.await(); // 等待其他线程到达屏障
System.out.println(Thread.currentThread().getName() + " 继续执行");
} catch (InterruptedException | BrokenBarrierException e) {
e.printStackTrace();
}
}
}
原理解析
CyclicBarrier 的内部实现基于 ReentrantLock 和 Condition,并通过一个 count 变量作为计数器来记录当前到达屏障的线程数。当计数器的值降为 0 时,表示所有线程都已到达屏障,此时屏障打开,所有线程继续执行。
构造方法
java
public CyclicBarrier(int parties) {
this(parties, null);
}
public CyclicBarrier(int parties, Runnable barrierAction) {
if (parties <= 0) throw new IllegalArgumentException();
this.parties = parties; // 初始化拦截线程数
this.count = parties; // 初始化计数器
this.barrierCommand = barrierAction; // 初始化屏障任务
}
await 方法
当线程调用 await() 方法时,实际上调用的是 dowait(false, 0L) 方法。await() 方法会将当前线程阻塞,直到所有线程都到达屏障。
java
public int await() throws InterruptedException, BrokenBarrierException {
try {
return dowait(false, 0L);
} catch (TimeoutException toe) {
throw new Error(toe); // 不会发生
}
}
dowait 方法
dowait 方法是 CyclicBarrier 的核心逻辑所在,负责实现线程的等待与屏障的打开。
java
private int dowait(boolean timed, long nanos)
throws InterruptedException, BrokenBarrierException, TimeoutException {
final ReentrantLock lock = this.lock;
lock.lock(); // 获取锁
try {
final Generation g = generation;
if (g.broken)
throw new BrokenBarrierException();
if (Thread.interrupted()) {
breakBarrier();
throw new InterruptedException();
}
int index = --count; // 递减计数器
if (index == 0) { // 当计数器为0,表示最后一个线程到达屏障
boolean ranAction = false;
try {
final Runnable command = barrierCommand;
if (command != null)
command.run(); // 执行屏障任务
ranAction = true;
nextGeneration(); // 开启下一轮
return 0;
} finally {
if (!ranAction)
breakBarrier();
}
}
// 继续等待
for (;;) {
try {
if (!timed)
trip.await();
else if (nanos > 0L)
nanos = trip.awaitNanos(nanos);
} catch (InterruptedException ie) {
if (g == generation && ! g.broken) {
breakBarrier();
throw ie;
} else {
Thread.currentThread().interrupt();
}
}
if (g.broken)
throw new BrokenBarrierException();
if (g != generation)
return index;
if (timed && nanos <= 0L) {
breakBarrier();
throw new TimeoutException();
}
}
} finally {
lock.unlock(); // 释放锁
}
}
实际应用场景
- 多线程任务的阶段性同步:在一些复杂的多线程任务中,需要各个线程在某些阶段进行同步,然后再继续下一阶段的执行。
- 并行计算的结果合并:多个线程进行并行计算,计算完成后需要将结果汇总,此时可以使用 CyclicBarrier 来等待所有计算线程完成后再进行汇总。
ReentrantLock 的概述与深入解析
ReentrantLock 是 Java 并发包(java.util.concurrent)中的一个重要类,用于实现显式锁(Explicit Lock)。它提供了比 synchronized 关键字更灵活和更强大的锁机制。在实际开发中,ReentrantLock 常用于替代 synchronized 关键字,以实现更复杂的同步需求。
基本用法
ReentrantLock 提供了显式的锁和解锁操作,需要手动进行锁的获取和释放。下面是一个简单的示例,展示了 ReentrantLock 的基本用法:
java
import java.util.concurrent.locks.ReentrantLock;
public class ReentrantLockExample {
private final ReentrantLock lock = new ReentrantLock();
public void performTask() {
lock.lock(); // 显式获取锁
try {
// 临界区代码
System.out.println(Thread.currentThread().getName() + " 正在执行任务");
} finally {
lock.unlock(); // 显式释放锁
}
}
public static void main(String[] args) {
ReentrantLockExample example = new ReentrantLockExample();
for (int i = 0; i < 5; i++) {
new Thread(example::performTask).start();
}
}
}
主要功能
- 可重入性:ReentrantLock 是可重入的,这意味着一个线程可以多次获取同一个锁而不会被自己阻塞,类似于 synchronized。每次调用 lock() 方法都会增加锁的重入次数,调用 unlock() 方法会减少锁的重入次数,直到重入次数为 0 时才真正释放锁。
- 公平锁和非公平锁:ReentrantLock 可以选择公平锁和非公平锁。公平锁(Fair Lock)按照线程请求锁的顺序来分配锁,而非公平锁(Non-Fair Lock)可能会让某些线程“插队”。默认情况下,ReentrantLock 是非公平锁,但可以通过构造函数指定为公平锁。
- 条件变量(Condition):ReentrantLock 提供了条件变量(Condition)来实现等待/通知机制,比 Object 的 wait 和 notify 更加灵活和强大。
可重入性示例
java
import java.util.concurrent.locks.ReentrantLock;
public class ReentrantLockReentrantExample {
private final ReentrantLock lock = new ReentrantLock();
public void outer() {
lock.lock();
try {
System.out.println(Thread.currentThread().getName() + " 进入外层方法");
inner();
} finally {
lock.unlock();
}
}
public void inner() {
lock.lock();
try {
System.out.println(Thread.currentThread().getName() + " 进入内层方法");
} finally {
lock.unlock();
}
}
public static void main(String[] args) {
ReentrantLockReentrantExample example = new ReentrantLockReentrantExample();
example.outer();
}
}
公平锁和非公平锁
java
import java.util.concurrent.locks.ReentrantLock;
public class FairLockExample {
private final ReentrantLock fairLock = new ReentrantLock(true); // 公平锁
private final ReentrantLock nonFairLock = new ReentrantLock(false); // 非公平锁
public void performTaskWithFairLock() {
fairLock.lock();
try {
System.out.println(Thread.currentThread().getName() + " 获取了公平锁");
} finally {
fairLock.unlock();
}
}
public void performTaskWithNonFairLock() {
nonFairLock.lock();
try {
System.out.println(Thread.currentThread().getName() + " 获取了非公平锁");
} finally {
nonFairLock.unlock();
}
}
public static void main(String[] args) {
FairLockExample example = new FairLockExample();
for (int i = 0; i < 5; i++) {
new Thread(example::performTaskWithFairLock).start();
new Thread(example::performTaskWithNonFairLock).start();
}
}
}
源码解析
ReentrantLock 的实现基于 AbstractQueuedSynchronizer(AQS),这是一个用于构建锁和同步器的框架。下面我们通过源码来深入解析 ReentrantLock 的实现。
构造函数
java
public ReentrantLock() {
sync = new NonfairSync();
}
public ReentrantLock(boolean fair) {
sync = fair ? new FairSync() : new NonfairSync();
}
ReentrantLock 通过 sync 变量来持有锁的具体实现,sync 可以是 FairSync(公平锁)或 NonfairSync(非公平锁)。
锁的获取
java
public void lock() {
sync.lock();
}
abstract static class Sync extends AbstractQueuedSynchronizer {
abstract void lock();
// 非公平锁的获取
static final class NonfairSync extends Sync {
final void lock() {
if (compareAndSetState(0, 1))
setExclusiveOwnerThread(Thread.currentThread());
else
acquire(1);
}
protected final boolean tryAcquire(int acquires) {
return nonfairTryAcquire(acquires);
}
}
// 公平锁的获取
static final class FairSync extends Sync {
final void lock() {
acquire(1);
}
protected final boolean tryAcquire(int acquires) {
final Thread current = Thread.currentThread();
int c = getState();
if (c == 0) {
if (!hasQueuedPredecessors() && compareAndSetState(0, 1)) {
setExclusiveOwnerThread(current);
return true;
}
} else if (current == getExclusiveOwnerThread()) {
int nextc = c + acquires;
if (nextc < 0) // overflow
throw new Error("Maximum lock count exceeded");
setState(nextc);
return true;
}
return false;
}
}
}
在 NonfairSync 中,lock() 方法首先尝试通过 CAS 操作直接获取锁,如果失败则调用 acquire(1),这是 AQS 提供的模板方法,用于获取独占锁。在 FairSync 中,直接调用 acquire(1) 来获取锁。
锁的释放
java
public void unlock() {
sync.release(1);
}
abstract static class Sync extends AbstractQueuedSynchronizer {
protected final boolean tryRelease(int releases) {
int c = getState() - releases;
if (Thread.currentThread() != getExclusiveOwnerThread())
throw new IllegalMonitorStateException();
boolean free = false;
if (c == 0) {
free = true;
setExclusiveOwnerThread(null);
}
setState(c);
return free;
}
}
unlock() 方法调用 sync.release(1) 来释放锁,release(1) 是 AQS 提供的模板方法,用于释放独占锁。在 tryRelease 方法中,通过减小锁的重入次数来释放锁,当重入次数为 0 时,真正释放锁。
具体应用场景
- 高度竞争的共享资源访问:在多个线程频繁竞争同一资源的情况下,使用 ReentrantLock 可以提供更高效的锁机制,特别是当需要实现公平性时。
java
import java.util.concurrent.locks.ReentrantLock;
public class SharedResource {
private final ReentrantLock lock = new ReentrantLock(true); // 公平锁
public void accessResource() {
lock.lock();
try {
// 资源访问操作
System.out.println(Thread.currentThread().getName() + " 正在访问资源");
} finally {
lock.unlock();
}
}
public static void main(String[] args) {
SharedResource resource = new SharedResource();
for (int i = 0; i < 10; i++) {
new Thread(resource::accessResource).start();
}
}
}
- 实现读写锁:虽然 Java 提供了专门的 ReadWriteLock 接口和 ReentrantReadWriteLock 实现,但我们也可以通过 ReentrantLock 自己实现一个简单的读写锁。
java
import java.util.concurrent.locks.ReentrantLock;
public class SimpleReadWriteLock {
private final ReentrantLock lock = new ReentrantLock();
private int readers = 0;
public void readLock() {
lock.lock();
try {
readers++;
} finally {
lock.unlock();
}
}
public void readUnlock() {
lock.lock();
try {
readers--;
if (readers == 0) {
lock.notifyAll();
}
} finally {
lock.unlock();
}
}
public void writeLock() {
lock.lock();
try {
while (readers > 0) {
lock.await();
}
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
}
}
public void writeUnlock() {
lock.unlock();
}
}
- 中断响应锁:ReentrantLock 提供了可中断的锁获取方法 lockInterruptibly(),允许线程在等待锁的过程中响应中断。
java
import java.util.concurrent.locks.ReentrantLock;
public class InterruptibleLockExample {
private final ReentrantLock lock = new ReentrantLock();
public void performTask() {
try {
lock.lockInterruptibly(); // 可中断的锁获取
try {
// 临界区代码
System.out.println(Thread.currentThread().getName() + " 正在执行任务");
Thread.sleep(2000); // 模拟长时间操作
} finally {
lock.unlock();
}
}
catch (InterruptedException e) {
System.out.println(Thread.currentThread().getName() + " 被中断");
}
}
public static void main(String[] args) {
InterruptibleLockExample example = new InterruptibleLockExample();
Thread t1 = new Thread(example::performTask);
Thread t2 = new Thread(() -> {
try {
Thread.sleep(1000);
t1.interrupt(); // 中断线程t1
} catch (InterruptedException e) {
e.printStackTrace();
}
});
t1.start();
t2.start();
}
}
在这个示例中,performTask 方法使用 lockInterruptibly() 获取锁,这允许线程在等待锁的过程中响应中断。当线程 t2 中断线程 t1 时,线程 t1 能够响应中断并退出锁的等待。
- 超时锁获取:ReentrantLock 提供了带超时的锁获取方法 tryLock(long timeout, TimeUnit unit),允许线程在指定时间内尝试获取锁,如果超时则返回 false。
java
import java.util.concurrent.TimeUnit;
import java.util.concurrent.locks.ReentrantLock;
public class TimeoutLockExample {
private final ReentrantLock lock = new ReentrantLock();
public void performTask() {
try {
if (lock.tryLock(2, TimeUnit.SECONDS)) { // 带超时的锁获取
try {
// 临界区代码
System.out.println(Thread.currentThread().getName() + " 获取到锁并执行任务");
} finally {
lock.unlock();
}
} else {
System.out.println(Thread.currentThread().getName() + " 获取锁超时");
}
} catch (InterruptedException e) {
System.out.println(Thread.currentThread().getName() + " 被中断");
}
}
public static void main(String[] args) {
TimeoutLockExample example = new TimeoutLockExample();
Thread t1 = new Thread(example::performTask);
Thread t2 = new Thread(example::performTask);
t1.start();
t2.start();
}
}
在这个示例中,performTask 方法使用 tryLock 方法尝试在 2 秒内获取锁,如果超时则打印超时信息。这避免了线程长时间等待锁的情况。
- 实现线程间的条件等待:ReentrantLock 提供了 Condition 对象,用于实现线程间的条件等待和通知机制。Condition 类似于 Object 的 wait 和 notify 方法,但提供了更灵活的功能,如指定多个条件变量。
java
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.ReentrantLock;
public class ConditionExample {
private final ReentrantLock lock = new ReentrantLock();
private final Condition condition = lock.newCondition();
private boolean ready = false;
public void awaitTask() {
lock.lock();
try {
while (!ready) {
System.out.println(Thread.currentThread().getName() + " 等待条件");
condition.await(); // 等待条件满足
}
System.out.println(Thread.currentThread().getName() + " 条件满足,继续执行");
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
lock.unlock();
}
}
public void signalTask() {
lock.lock();
try {
ready = true;
condition.signalAll(); // 通知所有等待的线程
System.out.println(Thread.currentThread().getName() + " 通知所有等待的线程");
} finally {
lock.unlock();
}
}
public static void main(String[] args) {
ConditionExample example = new ConditionExample();
Thread t1 = new Thread(example::awaitTask);
Thread t2 = new Thread(example::awaitTask);
Thread t3 = new Thread(example::signalTask);
t1.start();
t2.start();
try {
Thread.sleep(1000); // 确保t1和t2进入等待状态
} catch (InterruptedException e) {
e.printStackTrace();
}
t3.start();
}
}
在这个示例中,awaitTask 方法使用 condition.await() 进行条件等待,signalTask 方法使用 condition.signalAll() 通知所有等待的线程。这展示了 ReentrantLock 如何通过条件变量实现线程间的条件等待和通知机制。
总结
CyclicBarrier 和 ReentrantLock 是 Java 并发编程中非常重要的工具类。CyclicBarrier 主要用于多个线程在某个固定点上进行同步,常用于并行计算结果的汇总和阶段性同步。ReentrantLock 提供了比 synchronized 更灵活和强大的锁机制,适用于各种复杂的同步需求,如高度竞争的共享资源访问、实现读写锁、中断响应锁、超时锁获取以及条件变量等功能。
理解这些工具类的内部原理和使用方法,有助于我们在实际开发中更好地进行并发编程,避免一些常见的并发问题。希望通过本文的讲解,能让您对 CyclicBarrier 和 ReentrantLock 有一个更深入的理解,并在实际开发中灵活应用。
相关推荐
- Mac电脑强制删除任何软件方法-含自启动应用
-
对于打工者来说,进入企业上班使用的电脑大概率是会被监控起来,比如各种流行的数据防泄漏DLP,奇安信天擎,甚至360安全卫士,这些安全软件你想卸载是非常困难的,甚至卸载后它自己又安装回来了,并且还在你不...
- Linux基础知识 | 文件与目录大全讲解
-
1.linux文件权限与目录配置1.文件属性Linux一般将文件可存取的身份分为三个类别,分别是owner/group/others,且三种身份各read/write/execute等权限文...
- 文件保护不妥协:2025 年 10 款顶级加密工具推荐
-
数据安全无小事,2025年这10款加密工具凭借独特功能脱颖而出,从个人到企业场景全覆盖,第一款为Ping32,其余为国外英文软件。1.Ping32企业级加密核心工具,支持200+文件格...
- 省心省力 一个软件搞定系统维护_省心安装在哪里能找到
-
◆系统类似于我们居住的房间,需要经常打理才能保持清洁、高效。虽然它本身也自带一些清理和优化的工具,但借助于好用的第三方工具来执行这方面的任务,会更让人省心省力。下面笔者就为大家介绍一款集多项功能于一身...
- JAVA程序员常用的几个工具类_java程序员一般用什么软件写程序
-
好的工具做起事来常常事半功倍,下面介绍几个开发中常用到的工具类,收藏一下,也许后面真的会用到。字符串处理:org.apache.commons.lang.StringUtilsisBlank(Char...
- 手工解决Windows10的若干难题_windows10系统卡顿怎么解决
-
【电脑报在线】很多朋友已经开始使用Win10,估计还只是测试版本的原因,使用过程中难免会出现一些问题,这里介绍解决一些解决难题的技巧。技巧1:让ProjectSpartan“重归正途”从10074...
- System32文件夹千万不能删除,看完这篇你就知道为什么了
-
C:\Windows\System32目录是Windows操作系统的关键部分,重要的系统文件存储在该目录中。网上的一些恶作剧者可能会告诉你删除它,但你不应该尝试去操作,如果你尝试的话,我们会告诉你会发...
- Windows.old 文件夹:系统备份的解析与安全删除指南
-
Windows.old是Windows系统升级(如Win10升Win11)或重装时,系统自动在C盘创建的备份文件夹,其核心作用是保留旧系统的文件、程序与配置,为“回退旧系统”提供保...
- 遇到疑难杂症?Windows 10回收站问题巧解决
-
回收站是Windows10的一个重要组件。然而,我们在使用过程中,可能会遇到一些问题。例如,不论回收站里有没有文件,都显示同一个图标,让人无法判别回收站的空和满的真实情况;没有了像Windows7...
- 卸载软件怎么彻底删掉?简单几个步骤彻底卸载,电脑小白看过来
-
日常工作学习生活中,我们需要在安装一些软件程序,但随着软件的更新迭代速度,很多时候我们需要重新下载安装新的程序,这时就需要将旧的一些软件程序进行卸载。但是卸载软件虽然很简单,但是很多小伙伴们表示卸载不...
- 用不上就删!如何完全卸载OneDrive?
-
作为Windows10自带的云盘,OneDrive为资料的自动备份和同步提供了方便。然而,从隐私或其他方面考虑,有些人不愿意使用OneDrive。但Windows10本身不提供直接卸载OneDri...
- 【Linux知识】Linux下快速删除大量文件/文件夹方法
-
在Linux下,如果需要快速删除大量文件或文件夹,可以使用如下方法:使用rm命令删除文件:可以使用rm命令删除文件,例如:rm-rf/path/to/directory/*这个命令会递...
- 清理系统不用第三方工具_清理系统垃圾用什么软件
-
清理优化系统一定要借助于优化工具吗?其实,手动优化系统也没有那么神秘,掌握了方法和技巧,系统清理也是一件简单和随心的事。一方面要为每一个可能产生累赘的文件找到清理的方法,另一方面要寻找能够提高工作效率...
- 系统小技巧:软件卸载不了?这里办法多
-
在正常情况下,我们都是通过软件程序组中的卸载图标,或利用控制面板中的“程序和功能”模块来卸载软件的。但有时,我们也会发现利用卸载图标无法卸载软件或者卸载图标干脆丢失找不到了,甚至控制面板中卸载软件的功...
- 麒麟系统无法删除文件夹_麒麟系统删除文件权限不够
-
删除文件夹方法例:sudorm-rf文件夹名称。删除文件方法例:sudorm-r文件名包括扩展名。如果没有权限,给文件夹加一下权限再删。加最高权限chmod775文件名加可执行权限...
你 发表评论:
欢迎- 一周热门
- 最近发表
- 标签列表
-
- 如何绘制折线图 (52)
- javaabstract (48)
- 新浪微博头像 (53)
- grub4dos (66)
- s扫描器 (51)
- httpfile dll (48)
- ps实例教程 (55)
- taskmgr (51)
- s spline (61)
- vnc远程控制 (47)
- 数据丢失 (47)
- wbem (57)
- flac文件 (72)
- 网页制作基础教程 (53)
- 镜像文件刻录 (61)
- ug5 0软件免费下载 (78)
- debian下载 (53)
- ubuntu10 04 (60)
- web qq登录 (59)
- 笔记本变成无线路由 (52)
- flash player 11 4 (50)
- 右键菜单清理 (78)
- cuteftp 注册码 (57)
- ospf协议 (53)
- ms17 010 下载 (60)