多线程的锁,释放锁,重新获取锁的问题

23 篇文章 0 订阅
5 篇文章 0 订阅

线程通信-wait和notify方法介绍:

java.lang.Object类提供类两类用于操作线程通信的方法.

wait():执行该方法的线程对象释放同步锁,JVM把该线程存放到等待池中,等待其他的线程唤醒该线程.

notify:执行该方法的线程唤醒在等待池中等待的任意一个线程,把线程转到锁池中等待.

notifyAll():执行该方法的线程唤醒在等待池中等待的所有的线程,把线程转到锁池中等待.

注意:上述方法只能被同步监听锁对象来调用,否则报错IllegalMonitorStateException…


假设A线程和B线程共同操作一个X对象(同步锁),A,B线程可以通过X对象的wait和notify方法来进行通信,流程如下:

1:当A线程执行X对象的同步方法时,A线程持有X对象的锁,B线程没有执行机会,B线程在X对象的锁池中等待.

2:A线程在同步方法中执行X.wait()方法时,A线程释放X对象的锁,进入A线程进入X对象的等待池中.

3:在X对象的锁池中等待锁的B线程获取X对象的锁,执行X的另一个同步方法.

4:B线程在同步方法中执行X.notify()方法时,JVM把A线程从X对象的等待池中移动到X对象的锁池中,等待获取锁.

5:B线程执行完同步方法,释放锁.A线程获得锁,继续执行同步方法.
————————

wait() 导致当前线程等待,直到其他线程调用此对象的 notify()方法或 notifyAll()方法前,导致当前线程等待。
notify() 唤醒在此同步监视器上等待的单个线程,如果有多个线程在此同步监视器上等待,则会唤醒其中的一个,

功能代码
public class RoomMate implements Runnable{

Object cesuo=new Object();//临界资源厕所对象
public  void toBrush(){
	System.out.println(Thread.currentThread().getName()+"进去刷牙");
	
	try {
		Thread.sleep(2000);
	} catch (InterruptedException e) {
		// TODO Auto-generated catch block
		e.printStackTrace();
	}
	System.out.println(Thread.currentThread().getName()+"刷完牙出来了");
	
}

public  void toWC(){
	System.out.println(Thread.currentThread().getName()+"进去上厕所");
	
	try {
		Thread.sleep(2000);
	} catch (InterruptedException e) {
		// TODO Auto-generated catch block
		e.printStackTrace();
	}
	System.out.println(Thread.currentThread().getName()+"上完厕所出来了");
	
}

@Override
public void run() {
	// TODO Auto-generated method stub
	synchronized (cesuo) {
		if("张三".equals(Thread.currentThread().getName())){
			this.toBrush();
			try {
				cesuo.wait();//释放锁,张三进入等待状态
			} catch (InterruptedException e) {
				// TODO Auto-generated catch block
				e.printStackTrace();
			}
			this.toWC();
			cesuo.notify();//唤醒李四
		}else{
			this.toBrush();
			cesuo.notify();//唤醒张三
			try {
				cesuo.wait();//李四等待
			} catch (InterruptedException e) {
				// TODO Auto-generated catch block
				e.printStackTrace();
			}
			this.toWC();
			cesuo.notify();//唤醒李四
		}
	}
}

}
测试
public class Demo {

public static void main(String[] args) {
	Runnable r=new RoomMate();
	
	
	Thread t1=new Thread(r, "张三");
	
	Thread t2=new Thread(r, "李四");
	
	t1.start();
	t2.start();
}

}

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值