4_线程间定制化通信

Posted root_zhb

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了4_线程间定制化通信相关的知识,希望对你有一定的参考价值。

场景: A 线程打印 5 次 A,B 线程打印 10 次 B,C 线程打印 15 次 C,按照此顺序循环 10 轮
解决:给每个线程定义一个标志位,通过标志位完成定制化通信

编码流程
上锁
   判断条件(防止虚假唤醒),若不满足,conditionA.await();
   //具体的逻辑
   修改标志位
   通过 Condition 唤醒指定线程
解锁(finally)

class DemoClass{
    //通信对象:0--打印 A 1---打印 B 2----打印 C
    private int number = 0;
    //声明锁
    private Lock lock = new ReentrantLock();
    //声明钥匙 A
    private Condition conditionA = lock.newCondition();
    //声明钥匙 B
    private Condition conditionB = lock.newCondition();
    //声明钥匙 C
    private Condition conditionC = lock.newCondition();
    /**
     * A 打印 5 次
     */
    public void printA(int j){
        lock.lock();
        try {
            while (number != 0){
                conditionA.await();
            }
            System.out.println(Thread.currentThread().getName() + "输出 A,第" + j + " 轮开始");
            //输出 5 次 A
            for (int i = 0; i < 5; i++) {
                System.out.println("A");
            }
            //开始打印 B
            number = 1;
            //唤醒 B
            conditionB.signal();
        }catch (Exception e){
            e.printStackTrace();
        }finally {
            lock.unlock();
        }
    }
    /**
     * B 打印 10 次
     */
    public void printB(int j){
        lock.lock();
        try {
            while (number != 1){
                conditionB.await();
            }
            System.out.println(Thread.currentThread().getName() + "输出 B,第" + j + " 轮开始");
            //输出 10 次 B
            for (int i = 0; i < 10; i++) {
                System.out.println("B");
            }
            //开始打印 C
            number = 2;
            //唤醒 C
            conditionC.signal();
        }catch (Exception e){
            e.printStackTrace();
        }finally {
            lock.unlock();
        }
    }
    /**
     * C 打印 15 次
     */
    public void printC(int j){
        lock.lock();
        try {
            while (number != 2){
                conditionC.await();
            }
            System.out.println(Thread.currentThread().getName() + "输出 C,第" + j + " 轮开始");
            //输出 15 次 C
            for (int i = 0; i < 15; i++) {
                System.out.println("C");
            }
            System.out.println("-----------------------------------------");
            //开始打印 A
            number = 0;
            //唤醒 A
            conditionA.signal();
        }catch (Exception e){
            e.printStackTrace();
        }finally {
            lock.unlock();
        }
    }

    public static void main(String[] args) {
        DemoClass demoClass=new DemoClass();
        new Thread(()->{
            for (int i=0;i<10;i++){
                demoClass.printA(i);
            }
        },"A线程").start();

        new Thread(()->{
            for (int i=0;i<10;i++){
                demoClass.printB(i);
            }
        },"B线程").start();

        new Thread(()->{
            for (int i=0;i<10;i++){
                demoClass.printC(i);
            }
        },"C线程").start();
    }
}

以上是关于4_线程间定制化通信的主要内容,如果未能解决你的问题,请参考以下文章

线程间定制化通信---按顺序执行线程

JUC---06线程间通信

JUC---06线程间通信

阶段1 语言基础+高级_1-3-Java语言高级_05-异常与多线程_第4节 等待唤醒机制_5_线程间通信

多线程_线程间通信

3_线程间通信