多线程之wait和notify

目录

1.wait()方法

2. notify方法


因为线程之间是抢占式执行的,所以线程之间执行的先后顺序难以预知。但是实际开发中,我们希望线程之间的执行顺序是能被掌控的,比如线程2开始之前,需要线程1的某个任务先被执行。也就是说,很多时候,我们需要线程之间能协调工作,这就主要涉及到以下三个方法(均为Object类的方法):

  • wait():让当前线程进行等待状态
  • notify():唤醒在当前状态上等待的线程(只能唤醒一个等待该对象的对象锁的线程)
  • notifyAll():唤醒在当前状态上等待的线程(能唤醒所有等待该对象的对象锁的线程)

1.wait()方法

wait方法的作用是使当前线程进入阻塞等待.

当代码执行到wait()方法时,线程会经过以下过程:

  1. 当前线程释放锁.
  2. 使当前执行代码的线程进行等待(把线程放到等待队列中).
  3. 满足一定条件时线程被唤醒,重新尝试获取这个锁.

注意:wait()必须要和synchronized一起使用.因为wait()方法一执行就要先释放锁,所以wait必须要放到synchronized中使用.且synchronized锁的对象是和调用wait方法的对象是同一个对象.

wait使用示例:

运行这段代码:

 发现代码就停在这里了,这时我们需要使用另外一个唤醒的方法notify().让其能继续进行下去.

wait 结束等待的条件:

  1. 其他线程调用该对象的notify方法
  2. 其他线程调用该等待线程的interrupted方法,导致wait抛出InterruptedException异常
  3. wait等待的时间超时(wait方法重载了一个带有timeout参数的方法,来指定等待时间)

2. notify方法

notify方法的作用是唤醒等待的线程.

方法notify()要在同步方法或同步方法块中调用,该方法是用来通知那些正在等待该对象的对象锁的其他线程,对其发出通知notify,并使他们重新获取该对象的对象锁.

如果有多个线程都在等待,则线程调度器会随机挑选出一个呈wait状态的线程,没有先来后到这一说.

在notify方法后,当前线程不会马上释放该对象锁,要等到执行notify()方法的线程将程序执行完,也就是退出同步代码块之后才会释放对象锁.

使用notify方法唤醒线程:

分析以上代码,线程t1执行到locker.wait()时释放锁,阻塞等待;t2线程拿到锁,执行到locker.notify()时,将t1线程唤醒,但是此时t2线程还不会先将锁释放,会先将同步代码块执行完才会释放锁,之后t1线程再次开始执行.运行结果如下:

wait和notify方法要搭配使用,比如实现一个场景:线程1与线程2并发执行,线程1执行到某一条件时需要拿到线程2的结果才能继续往下执行,这时就可以用wait方法使其进入等待,当线程2将结果计算出来后,可用notify方法将等待该对象的对象锁的某一线程唤醒,使其重新拿到锁,继续执行.

notifyAll()方法:

notify方法只是唤醒一个线程,而notifyAll()方法可以一次唤醒所有的等待线程,比如有3个线程都在等待该对象的对象锁,那么notifyAll()方法可将这3个线程全部唤醒,但是这3个线程仍需要竞争这一把锁.

wait VS sleep

wait需要搭配synchronized使用,而sleep不需要;

wait是Object的方法,sleep是Thread的静态方法.

二者其实没有任何可比性而言.wait是用于线程之间的通信的,而sleep是让线程阻塞一段时间的.唯一的相同点就是都可以让线程停止执行一段时间.

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

哆啦A梦的110

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

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

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

打赏作者

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

抵扣说明:

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

余额充值