JUC实现精确顺序访问(线程通信例题)

在JUC编程中可以通过Condition类来进行精确阻塞与唤醒。

Condition类主要有两种方法:

1,Condition.await():使当前线程陷入阻塞,并释放锁。

2,  Condition.signal():唤醒该Condition对象所在的线程。

下面有个例题可以帮助理解。

题目要求:创建三个线程,线程1打印 " -A- " 5次 ,线程2打印" -B-" 10次,线程3打印 "-C-" 15次,并按照 A->B->C顺序进行打印,重复10轮。

以下是实现过程

package Condtion;

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

public class Wait_Demo {
	public static void main(String[] args) {
		Print print =new Print();

		//设置开始点——1对应A,2对应B,3对应C
		print.setflag(1);

		//lambda表达式,启动三个线程,为了方便描述,就命名它们分别为线程1,线程2,线程3
		new Thread(()->{
			try {
				for(int i=0;i<10;i++) {
					print.printA();
				}
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
		}).start();
		new Thread(()->{
			try {
				for(int i=0;i<10;i++)
					print.printB();
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
		}).start();
		new Thread(()->{
			try {
				for(int i=0;i<10;i++)
					print.printC();
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
		}).start();

	}
}

class Print{
	//锁对象为该类的对象
	private Lock lock=new ReentrantLock();

	//创建三个Condition类,它们是进行线程通信的核心
	private Condition ca=lock.newCondition();
	private Condition cb=lock.newCondition();
	private Condition cc=lock.newCondition();

	//flag可以设置起点
	private int flag;
	public  void setflag(int flag){
		this.flag=flag;
	}
	private  int count=0;

	public void printA() throws InterruptedException {
		lock.lock();//上锁
		try {
			//如果flag!=1则ca所在的线程阻塞,并释放锁
			while(flag!=1)//为了避免虚假唤醒需将if改成while。
				ca.await();
			System.out.println("第"+(++count)+"轮");
			for(int i=0;i<5;i++){
				System.out.print("-A-");
			}
			flag=2;//设置目标点为2
			cb.signal();//唤醒cb所在的线程.
		} finally {
			lock.unlock();//解锁
		}
	}public void printB() throws InterruptedException {
		lock.lock();
		try {
			//如果flag!=2则cb所在的线程阻塞,并释放锁
			while(flag!=2)
				cb.await();
			for(int i=0;i<10;i++){
				System.out.print("-B-");
			}
			flag=3;
			cc.signal();//唤醒cc所在的线程
		} finally {
			lock.unlock();
		}
	}public void printC() throws InterruptedException {
		lock.lock();
		try {
			//如果flag!=3则cc所在的线程阻塞,并释放锁
			while(flag!=3)
				cc.await();
			for(int i=0;i<15;i++){
				System.out.print("-C-");
			}
			System.out.println();
			flag=1;
			ca.signal();//唤醒ca所在的线程
		} finally {
			lock.unlock();
		}
	}
}

运行结果:

代码讲解:

创建三个线程(通过lambda表达式)使其处于就绪状态,每个线程分别调用资源类(Print)中的方法,因为三个线程均处于就绪状态,所以不确定谁先拿到锁,就会有三个情况。

情况1:假设线程2先拿到锁,那么其他线程则处于等待状态,线程2继续执行,此时由于设置的flag=1,那么就会调用await()方法,使线程2就会陷入阻塞同时释放锁,此时又会有两个情况:是线程1还是线程2拿到锁。因为flag=1,所以线程2拿到锁会陷入阻塞并释放锁,线程1拿到锁就继续向下执行,最后设置flag=2并唤醒Condition对象cb所在的线程2,但此时线程1仍然握着锁,所以线程2处于就绪状态。

线程1解锁后,此时线程2就会和线程3竞争锁,谁抢到锁是随机的,但是由于刚才设置了flag=2,就只有线程2拿到锁后才会执行,以此循环,达到 A->B->C顺序循环打印的效果。

情况2:线程3先拿到锁,过程同上......

情况3:线程1先拿到锁,由于初始设置的flag为1,线程1就不会陷入阻塞状态,继续执行下去,最后设置flag为2,然后执行signal方法,由于没有线程处于await状态所以这一步没有意义,但是不会对程序造成影响,线程1解锁后,线程2和线程3均有可能拿到锁,过程同情况1。

                                                                      -------萌新的Java学习博客,如果有错误欢迎指出。

  • 9
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

Z菌君

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

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

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

打赏作者

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

抵扣说明:

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

余额充值