从1.0版本开始,Java中的每一个对象都有一个内部锁。
将静态方法声明为synchronized也是合法的 ,如果调用这种方法,该方法获得相关的类对象的内部锁。
synchronized关键字经过编译之后,会在同步块的前后分别形成monitorenter和monitorexit这两个字节码指令,这两个字节码都需要一个reference类型的参数来指明要锁定和解锁的对象。如果Java程序中的synchronized明确指定了对象参数,那就是这个对象的reference;如果没有明确指定,那就根据synchronized修饰的是实例方法还是类方法,去取对应的对象实例或class对象来作为锁对象。
在虚拟机规范对monitorenter和monitorexit的行为描述中,有两点是需要特别注意到。
1. synchronized同步块对同一线程来说是可重入的,不会出现自己把自己锁死的问题。
2. 同步块在已进入的线程执行完之前,会阻塞后面其他线程的进入。
针对第一点的示例代码如下:
public classOne{
publicstaticvoidmain(String[] args)
{
Object lock = new Object();
synchronized(lock)
{
System.out.println("第1次进入锁定区域");
synchronized(lock)
{
System.out.println("第2次进入锁定区域");
}
}
}
}
参考《深入理解java虚拟机》一书