首先wait()是让线程进入到阻塞状态,notify()可以让阻塞的线程被唤醒,所以wait和notify必然是成对出现的。
如果一个线程被wait方法阻塞,那么必然需要另外一个线程通过notify方法来唤醒,从而去实现多个线程之间的通信,要实现多个线程之间的通信,除了使用管道流之外,只能去通过共享变量的方法来实现。也就是说线程t1修改了共享变量s,线程t2获得了修改后的共享变量的值,从而去完成数据的通信,但是多线程本身是具有并行执行的特点,所以在这种情况下线程t2在访问共享变量s之前要知道线程t1已经执行过了共享变量s,否则就需要等待,同时把处于等待下的线程被唤醒。所以想要实现线程间的通信,就必须要有一个静态条件,去控制线程什么时候条件等待,什么时候条件唤醒。
而synchronized的同步关键字就可以实现这样一个互斥的条件,也就是在通过共享变量来实现多个线程通信的场景下,参与通信的线程需要去竞争这个共享变量的一个锁资源,才能够有资格对共享变量进行修改,修改完后唤醒其他线程再次竞争同一个共享变量的锁,来获取修改后的数据从而去完成线程之间的一个通信。这也就是为什么wait和notify需要放在synchronized代码块里的原因。有了synchronized的同步锁就可以实现对于多个线程之间的一个互斥,从而去实现条件等待和条件唤醒。
为了避免wait和notify的错误使用,jdk强制要求把wait和notify写在同步代码块里否则会报出lllegal monitorState Exception的异常