后台开发笔记

为什么wait和notify要和Synchronized一起使用

2020-11-01  本文已影响0人  baifanger

1.正确的使用方式
线程间进行相互协作时,不可避免的会用到wait和notify。如下例子:

public static void testWait(){
        final Object obj = new Object();
        new Thread(){
            @Override
            public void run() {
                synchronized (obj){
                    try {
                        obj.wait();
                        System.out.println("thread 1");
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }

            }
        }.start();

        new Thread(){
            @Override
            public void run() {
                synchronized (obj){
                    System.out.println("thread 2");
                    obj.notify();
                }
            }
        }.start();
    }

结果:

thread 2
thread 1

上述代码可正常运行,但我们也发现了,在使用wait和notify时,必须在synchronized块或方法中,从官方文档我们也能看到样例:

 synchronized (obj) {
         while (<condition does not hold>)
             obj.wait(timeout, nanos);
         ... // Perform action appropriate to condition
     }

为什么要这样的,如果不用synchronized的话,会发生什么情况呢?

2.错误示例
将代码中的synchronized去除掉再运行会怎么样呢?

Exception in thread "Thread-0" Exception in thread "Thread-1" java.lang.IllegalMonitorStateException
    at java.lang.Object.notify(Native Method)
    at com.game.thread.ThreadTest$2.run(ThreadTest.java:32)
java.lang.IllegalMonitorStateException
    at java.lang.Object.wait(Native Method)
    at java.lang.Object.wait(Object.java:502)
    at com.game.thread.ThreadTest$1.run(ThreadTest.java:17)

运行结果,抛异常了。

3.原因
假设不会抛异常,那会发生什么样的情况呢?
当Thread1运行到obj.wait之前,由于时间片轮转,此时CPU调度到Thread2运行,然后obj.notify运行了。当再交由Thread1运行时,Thread1运行了wait,然后永远无法醒来了。
所以,在使用wait和notify时,他们必须在synchronized中,竞争同一对象锁。

4.释放锁的时机

5.wait,sleep,yield,join的区别

6.线程状态图

WX20201101-104332@2x.png

只有synchronized会让线程进入阻塞状态

上一篇 下一篇

猜你喜欢

热点阅读