Synchronization on a non-final field提示的缘由

引入

写代码的时候发现idea再sync代码块的位置出现这个提示,刚开始没有注意,后来出现问题了才知道其中的缘由。

看下面这段代码

public class T implements Runnable{
    Boolean b = new Boolean(false);

    public static void main(String[] args) throws InterruptedException {
        T t = new T();
        new Thread(t).start();
        Thread.sleep(1);
        new Thread(t).start();
    }

    @Override
    public void run() {
        synchronized (b){
            System.out.println(Thread.currentThread().getName());
            b = true;
            try {
                Thread.sleep(10000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}

按照我们的设想,每次进入sync代码块的应该只有一个线程,前后两个打印应该间隔10s左右,但是上面这段代码,就会出现两个线程都可以进入的情况,两个线程进入根本就没有间隔那么久,其根本原因就在于这个提示。

原因

之所以会出现这种情况,就在于我们改变了b所指向的对象。因为sync的原理在进行同步的时候是根据对象头的状态来进行同步的(这个不知道可以去查资料,这里不进行解释),我们改变了对象过的指向,当然就会导致这种情况。

你可能感兴趣的:(java多线程)