Java对象底层都关联monitor,使用synchronized时jvm会根据使用环境找到对象的monitor。根据monitor的状态加解锁的判断。如果成功加锁就成为monitor的唯一持有者。Monitor在被释放前不能被其他线程获取。
Synchronized在jvm编码后会产生monitorenter和monitorexit这两个字节码指令,获取和释放monitor。这两个字节码指令都需要引用类型的参数指明要锁定和解锁的对象,对于普通同步方法,锁是当前实例对象;对于静态同步方法,锁是当前类的class对象;对于同步方法块,锁是synchronized括号里的对象。
执行monitorenter指令时,首先尝试获取对象锁。如果这个对象没有被锁定,或当前线程已经持有锁,就把锁的计数器加1,执行monitorexit指令时将锁计数器减1。一旦计数器为0锁即被释放。