线程间通信方式

这篇博客探讨了如何使用synchronized关键字和Lock锁的Condition来实现三个线程交替打印'A'、'B'、'C',每打印100次。首先,展示了基于synchronized的简单实现,然后通过Lock锁的Condition进行了线程通信的优化,确保线程间的同步,并在打印完成后结束线程。在优化过程中,特别注意了线程结束时的信号传递,避免死锁和资源浪费。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

参考视频

实现一个场景

三个线程分别交替打印A、B、C,打印100次。

基于synchronized 锁的实现方式

public class test {

    private static Integer num = 0;
    public static void main(String[] args) {
        new Thread(new Runnable() {
            @Override
            public void run() {
                while (num < 100){
                    synchronized (num){
                        if (num%3==0&&num<=99){
                            System.out.println("A");
                            num++;
                        }
                    }
                }
            }
        }).start();

        new Thread(new Runnable() {
            @Override
            public void run() {
                while (num < 100){
                    synchronized (num){
                        if (num%3==1&&num<=99){
                            System.out.println("B");
                            num++;
                        }
                    }
                }
            }
        }).start();

        new Thread(new Runnable() {
            @Override
            public void run() {
                while (num < 100){
                    synchronized (num){
                        if (num%3==2&&num<=99){
                            System.out.println("C");
                            num++;
                        }
                    }
                }
            }
        }).start();

    }
}

基于Lock锁的Condition实现线程通信

package leetcode.leetcode11;

import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;

public class test06 {

    public static void main(String[] args) {
        Lock lock = new ReentrantLock();
        Condition conditionA = lock.newCondition();
        Condition conditionB = lock.newCondition();
        Condition conditionC = lock.newCondition();

        new Thread(new Runnable() {
            @Override
            public void run() {
                int a = 0;

                while (a<34){
                    lock.lock();
                    a++;
                    System.out.println("A");
                    conditionB.signal();

                    try {
                        conditionA.await();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    lock.unlock();
                }
            }
        }).start();
        new Thread(new Runnable() {
            @Override
            public void run() {
                int a = 0;
                while (a<33){
                    lock.lock();
                    a++;
                    System.out.println("B");
                    conditionC.signal();
                    try {
                        conditionB.await();
                        if (a==33) conditionC.signal();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    lock.unlock();
                }
            }
        }).start();
        new Thread(new Runnable() {
            @Override
            public void run() {
                int a = 0;
                while (a<33){
                    a++;
                    lock.lock();
                    System.out.println("C");
                    conditionA.signal();
                    try {
                        conditionC.await();
                        if (a==33) conditionA.signal();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    lock.unlock();
                }
            }
        }).start();
    }

}

注意这里面还是需要注意一下的,比如实现打印完成之后线程结束,这里面就要进行if判断了if (a==33) conditionA.signal();因为如果不判断的话第一个线程执行完之后唤醒第二个,第二个执行完结束了,没有办法唤醒第三个等待的线程了。

对第二种方式进行重构优化

package leetcode.leetcode11;

import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;

public class test06 {

    static class Runner implements Runnable{
        ReentrantLock lock;
        Condition now_condition;
        Condition next_condition;
        int num ;
        String ss;

        int a = 0;

        @Override
        public void run() {
            while (a<num){
                lock.lock();
                System.out.println(ss);
                a++;
                next_condition.signal();
                try {
                    now_condition.await();
                    if (a==33) next_condition.signal();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                lock.unlock();
            }
        }

        public Runner(ReentrantLock lock, Condition now_condition, Condition next_condition, int num, String ss) {
            this.lock = lock;
            this.now_condition = now_condition;
            this.next_condition = next_condition;
            this.num = num;
            this.ss = ss;
        }
    }
    public static void main(String[] args) {
        ReentrantLock lock = new ReentrantLock();
        Condition conditionA = lock.newCondition();
        Condition conditionB = lock.newCondition();
        Condition conditionC = lock.newCondition();

        Thread t1 = new Thread(new Runner(lock,conditionA,conditionB,34,"A"));
        Thread t2 = new Thread(new Runner(lock,conditionB,conditionC,33,"B"));
        Thread t3 = new Thread(new Runner(lock,conditionC,conditionA,33,"C"));

        t1.start();
        t2.start();
        t3.start();
    }

}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

北海冥鱼未眠

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值