锁定某对象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) {
}
}
}