乐趣区

wait-notify-notifyAll

wait notify notifyAll

wait(),notify(),notifyAll()都是 Object 的 final 方法,无法被重写。

  • 为什么是 Object 的方法

Java 的锁是对象级的而不是线程级,每个对象和类在逻辑上都是和一个监视器相关联的,线程为了进入临界区也就是同步块内,需要获得锁并等待锁可用,它们并不知道也不需要知道哪些线程持有锁,它们只需要知道当前资源是否被占用,是否可以获得锁,所以锁的持有状态应该由同步监视器来获取,而不是线程本身)

wait

  • wait()使当前线程阻塞,前提是,必须先获得锁,一般配合 synchronized 关键字使用,即,一般在 synchronized 同步代码块里使用 wait(),notify(),notifyAll()方法。
  • 由于 wait(),notify(),notifyAll()在 synchronized 代码块执行,说明当前线程一定是获取了锁的。
  • 当线程执行 wait()方法时候,会释放当前的锁,然后让出 CPU,进入等待状态。
  • 只有当 notify(),notifyAll()被执行时候,才会唤醒一个或多个正处于等待状态的线程,然后继续往下执行,直到执行完 synchronized 代码块的代码或是中途遇到 wait(),再次释放锁。
  • notify(),notifyAll()的执行只是唤醒沉睡的线程,而不会立即释放锁,锁的释放要看代码块的具体执行情况。所以在编程中,尽量在使用了 notify(),notifyAll()后立即退出临界区,以唤醒其他线程让其获得锁。
  • wait() 需要被 try catch 包围,以便发生异常中断也可以使 wait 等待的线程唤醒。
  • notify()和 wait()的顺序不能错,如果 A 线程先执行 notify()方法,B 线程再执行 wait()方法,那么 B 线程是无法被唤醒的。

notify & notifyAll 区别

notify()方法只唤醒一个等待(对象的)线程并使该线程开始执行。所以如果有多个线程等待一个对象,这个方法只会唤醒其中一个线程,选择哪个线程取决于操作系统对多线程管理的实现。notifyAll() 会唤醒所有等待 (对象的) 线程,尽管哪一个线程将会第一个处理取决于操作系统的实现。如果当前情况下有多个线程需要被唤醒,推荐使用 notifyAll()方法。比如在生产者 - 消费者里面的使用,每次都需要唤醒所有的消费者或是生产者,以判断程序是否可以继续往下执行。

退出移动版