• Java 线程工具类 加法/减法/信号量


    参考

    描述(JDK文档内容)

    1. CountDownLatch 减法计数器:允许一个或多个线程等待直到在其他线程中执行的一组操作完成的同步辅助。(通过 countDow() 方法进行减少计数,当计数为0时 await 处会被唤醒)

      • 构造方法 :CountDownLatch(int count) 构造一个用给定计数初始化的 CountDownLatch
      • countDown() :减少锁存器的计数,如果计数达到零则释放所有等待的线程
      • await():导致当前线程等到锁存器倒计数到零,除非线程是 interrupted
    2. CyclicBarrier 加法计数器:一种同步辅助工具,允许一组线程全部等待彼此到达公共障碍点。(通过 await() 方法进行增加计数,当计数为 parties 时执行 barrierAction 代码。)

      • CyclicBarrier(int parties, Runnable barrierAction):创建一个新的 CyclicBarrier ,当给定数量的参与方(线程)等待它时将跳闸,并且当屏障被触发时执行给定的屏障操作,由进入屏障的最后一个线程执行。(有两个构造方法,第一个构造方法可以不传入 Runnable/Lambda,而不执行指定操作 )
      • await():等待所有 parties 在此障碍上调用 await
      • reset():将屏障重置为其初始状态。
    3. Semaphore 信号量:计数信号量。 从概念上讲,信号量保持一组许可。 如果有必要,每个 acquire() 都会阻止,直到有许可证,然后接受。 每个 release() 都添加了许可证,可能会释放阻止收购者。 但是,没有使用实际的许可对象; Semaphore 只保留可用数量并相应地采取行动。(通过 acquire 获取一个执行许可,当执行许可被授权完毕之后,后面还需要获取授权的线程会堵塞并等待;当线程执行完毕后,通过 release 归还许可。)

      • Semaphore(int permits):使用给定数量的许可和非公平公平设置创建 Semaphore 。(第二个构造方法需要传入一个参数来设置 是否公平锁,默认不传入是非公平锁)
      • acquire():从此信号量获取许可,阻止直到一个可用,或者线程为 interrupted 。
      • release():发布许可证,将其返回到信号量。

    代码(狂神说代码)

    • CountDownLatch 减法计数器
    package thread;
    
    import java.util.*;
    import java.util.concurrent.CopyOnWriteArrayList;
    import java.util.concurrent.CopyOnWriteArraySet;
    import java.util.concurrent.CountDownLatch;
    import java.util.concurrent.TimeUnit;
    
    /**
     * @Author 夏秋初
     * @Date 2022/3/1 15:36
     */
    public class Test {
        public static void main(String[] args) throws InterruptedException {
            // 减法计数器,初始化参数为设置计数次数
            CountDownLatch countDownLatch = new CountDownLatch(10);
            //
            for (int i = 0; i < 10; i++) {
                new Thread(()->{
                    try {
                        TimeUnit.SECONDS.sleep(3);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    System.out.println(Thread.currentThread().getName()+"执行完毕");
                    // 计数减一
                    countDownLatch.countDown();
                }, String.valueOf(i)).start();
            }
            // 计数器清零前不会结束
            countDownLatch.await();
            System.out.println("程序加载完毕");
        }
    }
    

    image

    • CyclicBarrier 加法计数器
    package thread;
    
    import java.util.*;
    import java.util.concurrent.*;
    
    /**
     * @Author 夏秋初
     * @Date 2022/3/1 15:36
     */
    public class Test {
        public static void main(String[] args) throws InterruptedException {
            // 加法计数器
            CyclicBarrier cyclicBarrier = new CyclicBarrier(10, ()->{
                System.out.println("程序执行完毕");
            });
            //
            for (int i = 0; i < 10; i++) {
                new Thread(()->{
                    try {
                        TimeUnit.SECONDS.sleep(3);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    System.out.println(Thread.currentThread().getName()+"执行完毕");
                    // 计数加一
                    try {
                        cyclicBarrier.await();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    } catch (BrokenBarrierException e) {
                        e.printStackTrace();
                    }
                }, String.valueOf(i)).start();
            }
        }
    }
    
    

    image

    • Semaphore 信号量
    package thread;
    
    import java.util.*;
    import java.util.concurrent.*;
    
    /**
     * @Author 夏秋初
     * @Date 2022/3/1 15:36
     */
    public class Test {
        public static void main(String[] args) throws InterruptedException {
            // 限流
            Semaphore semaphore = new Semaphore(2);
            //
            for (int i = 0; i < 10; i++) {
                new Thread(()->{
                    try {
                        semaphore.acquire();
                        System.out.println(Thread.currentThread().getName()+"执行完毕");
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }finally {
                        semaphore.release();
                    }
                }, String.valueOf(i)).start();
            }
        }
    }
    
    

    image

  • 相关阅读:
    [leetcode]Reverse Words in a String
    *[topcoder]ChooseTheBestOne
    priority_queue c++
    [topcoder]BoxesDiv2
    *[topcoder]IncrementingSequence
    [topcoder]HappyLetterDiv2
    c++ stl 使用汇总(string,vector,map,set)
    2014找工作总结-机会往往留给有准备的人 (转,后台开发面经)
    2015互联网校招总结—一路走来 (移动端开发面经,另有面经汇总)
    通过金矿模型介绍动态规划(转)
  • 原文地址:https://www.cnblogs.com/xiaqiuchu/p/15951059.html
Copyright © 2020-2023  润新知