Java的高并发编程系列(三)

时间:2021-11-22 15:25:51

锁定某对象o,如果o的属性发生改变,不影响锁的使用,但是如果o变成另外一个对象,则锁定的对象发生改变,应该避免将锁定对象的引用变成另外一个对象。

public class Demo17 {

    Object o = new Object();

    public void test(){
        synchronized (o) {
            while (true) {
                try {
                    TimeUnit.SECONDS.sleep(1);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println(Thread.currentThread().getName());

            }
        }
    }

    public static void main(String[] args) {
        Demo17 demo17 = new Demo17();
        //启动第一个线程
        new Thread(demo17 :: test, "t1").start(); //JDK1.8新特性

        try {
            TimeUnit.SECONDS.sleep(3);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }

        //启动第二个线程
        Thread t2 = new Thread(demo17 :: test, "t2");

        demo17.o = new Object(); //所对象发生改变,所以t2线程得以执行,如果注释掉这句话,线程t2将永远得不到执行机会

        t2.start();
    }
}

不要以字符串常量作为锁定的对象
在下面的例子中,test1和test2其实锁定的是同一个对象
这种情况还会发生比较诡异的现象,比如你用到了一个类库,在该类库中代码锁定了字符串”hello”,
但是你读不到源码,所以你在自己的代码中也锁定了”hello”,这时候就有可能发生非常诡异的死锁阻塞, 因为你的程序和你用的的类库不经意间使用了同一把锁。

public class Demo18 {

    String s1 = "hello";
    String s2 = "hello";

    public void test1(){
        synchronized (s1) {

        }
    }

    public void test2(){
        synchronized (s2) {

        }
    }
}