锁 - 线程屏障CyclicBarrier

CyclicBarrier 直译是:循环屏障、可重用屏障。日常交流过程中,也可能称作:线程屏障。

跟信号量、倒数锁一样,都可以简单地实现“多个子线程执行完毕,主线程开始执行”的功能。

而屏障锁是其中较为特殊的:

  • 几个线程互相等待,直到某个公共屏障点 (common barrier point),它们最终会同时完成任务。
  • 名字中的Cyclic直译为循环,他有着可循环使用的特点(可重用)。
  • 对于失败的同步,CyclicBarrier 使用了一种要么全部要么全不 (all-or-none) 的破坏模式,在某个线程异常之后,全部线程将同时失败。

demo

主线程需要5个并发的初始化操作,5个线程全部执行完毕,主线程开始执行。

/**
 * @author ChenSS on 2018年2月3日
 */
public class Test {
    // 注意这里参与线程数量要改成6,主线程和子线程一起等待
    private static CyclicBarrier latch = new CyclicBarrier(6);
    
    public static void main(String[] args) throws InterruptedException, BrokenBarrierException {
        int count = 5;
        while (count-- > 0) {
            final int n = count;
            new Thread(() -> {
                try {
                    Thread.sleep(3000 + RandomUtils.nextLong(3000));
                    System.out.println("完成第" + n + "个步骤");
                    latch.await();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                } catch (BrokenBarrierException e) {
                    e.printStackTrace();
                }
            }).start();
        }
        // 希望CyclicBarrier处理倒计时锁的功能,主线程需要和子线程一起进入等待,共同到达初始化完成的屏障点
        latch.await();
        // 引用置空
        latch = null;
        System.out.println("五个初始化操作成功,进行别的操作!");
        
    }
}

官方Demo

收录于 jdk 官方文档中的一段代码

在这个例子中,每个 worker 线程处理矩阵的一行,在处理完所有的行之前,该线程将一直在屏障处等待。

所有的子线程执行结束之后,执行所提供的 Runnable 屏障操作,总共重复执行了5次之后退出程序。

import java.util.concurrent.BrokenBarrierException;
import java.util.concurrent.CyclicBarrier;

class Solver {
    final int N;
    final float[][] data;
    final CyclicBarrier barrier;
    
    // 设置一个退出标志位
    boolean done = true;
    // 循环5此之后,将done设置为false,退出循环
    int time = 5;
    
    // 每个Runnable负责处理一行的数据
    class Worker implements Runnable {
        int myRow;

        Worker(int row) {
            myRow = row;
        }

        public void run() {
            while(done){
                try {
                    for (float fs : data[myRow]) {
                        System.out.println(Thread.currentThread().getName() + " :" + fs);
                    }
                    barrier.await();
                } catch (InterruptedException ex) {
                    return;
                } catch (BrokenBarrierException e) {
                    e.printStackTrace();
                }
            }
        }
    }

    public Solver(float[][] matrix) {
        data = matrix;
        N = matrix.length;
        
        barrier = new CyclicBarrier(N, () -> {
            // 每一轮循环结束之后的回调函数
            if(time -- != 0){
                System.out.println("OK");
            } else {
                done = false;
            }
        });
        for (int i = 0; i < N; ++i) {
            new Thread(new Worker(i)).start();
        }
    }
}

public class Test {
    public static void main(String[] args) throws InterruptedException {
        float[][] data = { { 1f, 2f }, { 3f, 4f } };
        new Solver(data);
    }
}

posted on   疯狂的妞妞  阅读(249)  评论(0编辑  收藏  举报

(评论功能已被禁用)
编辑推荐:
· 开发者必知的日志记录最佳实践
· SQL Server 2025 AI相关能力初探
· Linux系列:如何用 C#调用 C方法造成内存泄露
· AI与.NET技术实操系列(二):开始使用ML.NET
· 记一次.NET内存居高不下排查解决与启示
阅读排行:
· 开源Multi-agent AI智能体框架aevatar.ai,欢迎大家贡献代码
· Manus重磅发布:全球首款通用AI代理技术深度解析与实战指南
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· 没有Manus邀请码?试试免邀请码的MGX或者开源的OpenManus吧
· 园子的第一款AI主题卫衣上架——"HELLO! HOW CAN I ASSIST YOU TODAY
< 2025年3月 >
23 24 25 26 27 28 1
2 3 4 5 6 7 8
9 10 11 12 13 14 15
16 17 18 19 20 21 22
23 24 25 26 27 28 29
30 31 1 2 3 4 5

导航

统计

点击右上角即可分享
微信分享提示