java并发编程之CountDownLatch,CyclicBarrier,Semaphore的简单应用

前言

这三个类都是在java1.5的时候由Doug Lea大神添加于java.util.concurrent,这三个辅助类都基于AQS同步器框架实现,下面我们简单介绍下它们的简单使用

CountDownLatch

CountDownLatch类似是一个计数器,他可以实现需要所有任务都执行完毕才可以执行接下来的任务,日常场景中我们可以使用他来做并行分布运算,借用多核cpu对数据分别进行计算,然后再汇总,也可以实现在加载某些东西前初始化一些信息。

主要方法

public CountDownLatch(int count);//构造函数
public void countDown();//计数器-1
public void await() throws InterruptedException;//挂起
public boolean await(long timeout, TimeUnit unit);//与await()类似,这里可以指定时间,达到时间如果计数器没归0也可以执行下面的东西

简单例子

private static void countDownLatchApply() throws Exception {
        //初始化2个
        CountDownLatch countDownLatch = new CountDownLatch(2);
        new Thread(() -> {
            try {
                System.out.println("我是线程" + Thread.currentThread().getName() + "我执行在"
                        +LocalDateTime.now().format(DateTimeFormatter.ofPattern("yyyy/MM/dd HH:mm:ss")));
                //模拟处理耗时
                Thread.sleep(2000L);
            } catch (InterruptedException e) {
                e.printStackTrace();
            } finally {
                countDownLatch.countDown();
            }
        }).start();

        new Thread(() -> {
            try {
                System.out.println("我是线程" + Thread.currentThread().getName() + "我执行在"+
                        LocalDateTime.now().format(DateTimeFormatter.ofPattern("yyyy/MM/dd HH:mm:ss")));
                //模拟处理耗时
                Thread.sleep(2000L);
            } catch (InterruptedException e) {
                e.printStackTrace();
            } finally {
                countDownLatch.countDown();
            }
        }).start();


        countDownLatch.await();

        new Thread(() -> {
            System.out.println("我是线程" + Thread.currentThread().getName() + "我需要前面两个执行完我才可以执行,我执行在"+
                    LocalDateTime.now().format(DateTimeFormatter.ofPattern("yyyy/MM/dd HH:mm:ss")));
        }).start();

    }

执行结果

我是线程Thread-1我执行在2019/05/15 08:05:36
我是线程Thread-0我执行在2019/05/15 08:05:36
我是线程Thread-2我需要前面两个执行完我才可以执行,我执行在2019/05/15 08:05:38

我们会看到线程2是需要在线程0和1执行玩后再执行的,我特意让线程休眠了2秒,后面的时间也印证了线程2是在线程0和线程1后2秒执行的

CyclicBarrier

回环栅栏,他可以使线程全部到达一个同步点后,再一起执行下面的动作,他是可重用的,等线程到达同步点,这个线程是可以被做其他使用的,我们姑且叫这个状态为可重用态,当调用await(),线程就为可重用态

主要方法

public CyclicBarrier(int parties);//构造方法
public CyclicBarrier(int parties, Runnable barrierAction);//构造方法,可实现更复杂的动作
public int await() throws InterruptedException, BrokenBarrierException;//挂起
public int await(long timeout, TimeUnit unit);//带时间,到期可执行下面操作

简单例子

 private static void cyclicBarrierApply() {

        CyclicBarrier cyclicBarrier=new CyclicBarrier(4, new Runnable() {
            @Override
            public void run() {
                System.out.println("我是线程"+Thread.currentThread().getName()+",老板,我按照你的要求在他们向你汇报前检查了他们4个的工作,他们一会会亲自向你汇报");
            }
        });

        for (int i = 0; i <4 ; i++) {
            new Thread(() ->{
                try {
                    Thread.sleep(2000L);
                    System.out.println("我是线程"+Thread.currentThread().getName()+",我的工作做完了,等其他线程工作好"
                    +",我们一起去向老板汇报");
                    cyclicBarrier.await();
                } catch (Exception e) {
                    e.printStackTrace();
                }finally {
                    System.out.println("老板,我来向你汇报工作了");
                }

            }).start();
        }

    }

执行结果

我是线程Thread-0,我的工作做完了,等其他线程工作好,我们一起去向老板汇报
我是线程Thread-1,我的工作做完了,等其他线程工作好,我们一起去向老板汇报
我是线程Thread-3,我的工作做完了,等其他线程工作好,我们一起去向老板汇报
我是线程Thread-2,我的工作做完了,等其他线程工作好,我们一起去向老板汇报
我是线程Thread-2,老板,我按照你的要求在他们向你汇报前检查了他们4个的工作,他们一会会亲自向你汇报
我是线程Thread-2老板,我来向你汇报工作了
我是线程Thread-3老板,我来向你汇报工作了
我是线程Thread-1老板,我来向你汇报工作了
我是线程Thread-0老板,我来向你汇报工作了

我们可以看到所有线程最开始都在完成自己的工作,并等待其他线程一起向“老板”汇报,线程完成工作了,变为了可重用态,这个时候线程2被重用,他被“老板”安排在四个线程汇报工作前先检查他们的工作,也就是第二个构造方法的运用,最后四个线程一起去向“老板”汇报工作。

CountDownLatch与CyclicBarrier的比较

他们的功能有一些类似,CountDownLatch是所有线程都到达一个点才能执行下面的动作,而CyclicBarrier是所有线程都到达一个点再一起执行下面的动作,CountDownLatch不可被重用,CyclicBarrier可以被重用

Semaphore

信号量,它可用于对资源进行有效的控制,获取到许可就可以使用,使用完许可主动释放掉,获取不到就需要等到有许可可以使用

主要方法

public Semaphore(int permits) //参数表示许可数目,同时可以允许多少线程进行访问
public Semaphore(int permits, boolean fair)  //fair表示是否是公平的,等待时间越久的越先获得许可
public void acquire() throws InterruptedException ;    //获取一个许可 会造成阻塞
public void acquire(int permits) throws InterruptedException ;    //获取permits个许可 会造成阻塞
public void release() ;        //释放一个许可 会造成阻塞
public void release(int permits) ;   //释放permits个许可 会造成阻塞
//下面四个方法与上面的一样,但是下面会立即获得结果,不会阻塞,操作失败就返回false
public boolean tryAcquire();
public boolean tryAcquire(long timeout, TimeUnit unit) throws InterruptedException
public boolean tryAcquire(int permits)
public boolean tryAcquire(int permits, long timeout, TimeUnit unit)

简单例子

private static void semaphoreApply() throws Exception {
        //仓库管理员
        Semaphore semaphore = new Semaphore(4);
        for (int i = 0; i < 10; i++) {
            new Thread(() -> {
                try {
                    //向仓库管理员要一把钥匙
                    semaphore.acquire();
                    System.out.println("我是线程" + Thread.currentThread().getName() + "我成功申请到了一把钥匙");
                    Thread.sleep(3000L);
                    System.out.println("我是线程" + Thread.currentThread().getName() + "我使用好了钥匙");
                    //给管理员说我用好了
                    semaphore.release();
                    System.out.println("我是线程" + Thread.currentThread().getName() + "我已经将钥匙还回去了");
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }).start();
        }

    }

运行结果

我是线程Thread-0我成功申请到了一把钥匙
我是线程Thread-2我成功申请到了一把钥匙
我是线程Thread-1我成功申请到了一把钥匙
我是线程Thread-3我成功申请到了一把钥匙
我是线程Thread-2我使用好了钥匙
我是线程Thread-1我使用好了钥匙
我是线程Thread-4我成功申请到了一把钥匙
我是线程Thread-6我成功申请到了一把钥匙
我是线程Thread-0我使用好了钥匙
我是线程Thread-2我已经将钥匙还回去了
我是线程Thread-1我已经将钥匙还回去了
我是线程Thread-5我成功申请到了一把钥匙
我是线程Thread-0我已经将钥匙还回去了
我是线程Thread-3我使用好了钥匙
我是线程Thread-3我已经将钥匙还回去了
我是线程Thread-7我成功申请到了一把钥匙
我是线程Thread-5我使用好了钥匙
我是线程Thread-5我已经将钥匙还回去了
我是线程Thread-6我使用好了钥匙
我是线程Thread-7我使用好了钥匙
我是线程Thread-9我成功申请到了一把钥匙
我是线程Thread-4我使用好了钥匙
我是线程Thread-7我已经将钥匙还回去了
我是线程Thread-6我已经将钥匙还回去了
我是线程Thread-8我成功申请到了一把钥匙
我是线程Thread-4我已经将钥匙还回去了
我是线程Thread-9我使用好了钥匙
我是线程Thread-9我已经将钥匙还回去了
我是线程Thread-8我使用好了钥匙
我是线程Thread-8我已经将钥匙还回去了

我们可以看到,最开始只有4个获取到了许可,我在这里将线程休眠3秒,模拟耗时,在这期间没有其他线程获取到许可,并且同时也只能有四个在运行,每下一个获取到许可都必须是有线程释放许可,

参考资料:《java编程思想》

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,968评论 6 482
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,601评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 153,220评论 0 344
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,416评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,425评论 5 374
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,144评论 1 285
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,432评论 3 401
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,088评论 0 261
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,586评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,028评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,137评论 1 334
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,783评论 4 324
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,343评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,333评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,559评论 1 262
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,595评论 2 355
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,901评论 2 345

推荐阅读更多精彩内容