线程通信的例子:使用两个线程打印 1-100。线程1, 线程2 交替打印

  • 涉及到的三个方法
  • wait():一旦执行此方法,当前线程就进入阻塞状态,并释放同步监视器
  • notify():一旦执行此方法,就会唤醒被阻塞的进程,如果有多个被wait(),就唤醒优先级最高的
  • notifyAll():一旦执行此方法,就会唤醒所有被阻塞的进程
  • 说明:
  • 1、wait(),notify(),notifyAll()三个方法必须使用在同步代码块或同步方法中
  • 2、wait(),notify(),notifyAll()三个方法的调用者必须是同步代码块或同步方法中的同步监视器,否则,会出现IllegalMonitorStateException异常
  • 3、wait(),notify(),notifyAll()三个方法是定义在java.lang.Object类中。
  • sleep() 和 wait()的异同?
  • 1、相同点:一旦执行方法,都可以使得当前的线程进入阻塞状态。
  • 2、不同点:
  • 1)两个方法声明的位置不同:Thread类中声明sleep() , Object类中声明wait()
  • 2)调用的要求不同:sleep()可以在任何需要的场景下调用。 wait()必须使用在同步代码块或同步方法中
  • 3)关于是否释放同步监视器:如果两个方法都使用在同步代码块或同步方法中,sleep()不会释放锁,wait()会释放锁。
class Number implements Runnable{
    private int number = 1;
    private final Object obj = new Object();
    @Override
    public void run() {

        while(true){

            synchronized (obj) {//this

                obj.notify();//this.notify()

                if(number <= 100){

                    try {
                        Thread.sleep(10);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }

                    System.out.println(Thread.currentThread().getName() + ":" + number);
                    number++;

                    try {
                        //使得调用如下wait()方法的线程进入阻塞状态
                        obj.wait();//this.wait()
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }

                }else{
                    break;
                }
            }

        }

    }
}


public class CommunicationTest {
    public static void main(String[] args) {
        Number number = new Number();
        Thread t1 = new Thread(number);
        Thread t2 = new Thread(number);

        t1.setName("线程1");
        t2.setName("线程2");

        t1.start();
        t2.start();
    }
}