在并发编程中,synchronized一直是被使用非常频繁的,很多人会把它称为重量级锁,但是,JavaSE1.6对它有一个重大优化,在1.6中为了减少获得锁和释放锁带来的性能消耗而引入了偏向锁和轻量级锁。
我们知道Java中的每一个对象都可以作为锁,具体表现为三种形式:
1.对于普通同步方法,锁是当前实例对象
2.对于静态同步方法,锁是当前类的Class对象
3.对于同步代码块,锁是synchronized括号中配置的对象
我们知道,当一个线程试图访问同步代码块或方法时,它首先要得到锁,退出或者抛出异常时必须释放锁,那么我们所说的锁到底存放在哪里?锁中会存放什么信息呢?
synchronized锁的实现原理
在JVM规范中,任何一个对象都与一个Monitor与之关联,当且一个Monitor被持有后,它将处于锁定状态。Synchronized在JVM里的实现都是基于进入和退出Monitor对象来实现方法同步和代码块同步,虽然具体实现细节不一样,但是都可以通过成对的MonitorEnter和MonitorExit指令来实现。MonitorEnter指令插入在同步代码块的开始位置,当代码执行到该指令时,将会尝试获取该对象Monitor的所有权,即尝试获得该对象的锁,而monitorExit指令则插入在方法结束处和异常处,JVM保证每个MonitorEnter必须有对应的MonitorExit。
也就是说,在并发的环境下,假如有一个线程A要执行同步代码块,该同步块括号中配置的对象对应了一个Monitor,当该线程执行到MonitorEnter指令时