线程之间的通信

wait() 与 notify() 和 notifyAll()

  • wait():令当前线程挂起并放弃CPU、同步资源并等待,使别的线程可访问并修改共享资源,而当前线程排队等候其他线程调用notify()或notifyAll()方法唤醒,唤醒后等待重新获得对监视器的所有权后才能继续执行。
  • notify():唤醒正在排队等待同步资源的线程中优先级最高者结束等待
  • notifyAll ():唤醒正在排队等待资源的所有线程结束等待.

这三个方法只有在synchronized方法或synchronized代码块中才能使用,否则会报java.lang.IllegalMonitorStateException异常。

因为这三个方法必须有锁对象调用,而任意对象都可以作为synchronized的同步锁,因此这三个方法只能在Object类中声明。

sleep() 和 wait()的异同?

1.相同点:一旦执行方法,都可以使得当前的线程进入阻塞状态。

2.不同点:

​ 1)两个方法声明的位置不同:Thread类中声明sleep() , Object类中声明wait()

​ 2)调用的要求不同:sleep()可以在任何需要的场景下调用。 wait()必须使用在同步代码块或同步方法中

​ 3)关于是否释放同步监视器:sleep()不会释放锁,wait()会释放锁。

例题使用两个线程印 打印 1-100 。线程1, 线程2 交替打印

class Number implements Runnable
{
    private static int n = 1;
    @Override
    public void run() {
         while (true) {
             synchronized (Number.class) {
                 Number.class.notify();
                 if (n <= 100) {
                     System.out.println(Thread.currentThread().getName() + ":" + n++);
                     try {
                         Number.class.wait();
                     } catch (InterruptedException e) {
                         e.printStackTrace();
                     }
                 } else {
                     break;
                 }
             }

         }
    }
}

public class Communication {

    public static void main(String[] args) {
        Number number1 = new Number();

        new Thread(number1).start();
        new Thread(number1).start();
    }
}