简介
synchronized可以锁在方法上也可以锁在代码块上
对非静态的方法加锁时,锁的是对象,称为对象锁。
对静态的方法加锁时,称为类锁。
原理
1)synchronized同步代码块的实现是通过monitorenter和monitorexit指令,monitorenter标识开始位置,monitorexit标识结束位置。monitorenter编译后插入同步代码块开始位置,monitorexit插入方法结束处和异常处。当执行monitorenter指令时,线程试图获取锁也就是获取monitor(monitor对象存在于每个]ava对象的对象头中,synchronized锁便是通过这种方式获取锁的,也是为什么Java中任意对象可以作为锁的原因)的持有权。
其内部包含一个计数器,当计数器为0则可以成功获取,获取后将锁计数器设为1。相应的在执行monitorexit指令后,将锁计数器设为0,表明锁被释放。如果获取对象锁失败,那当前线程就要阻塞等待,直到锁被另外一个线程释放为止。
2)synchronized修饰的方法并没有monitorenter指令和monitorexit指令,取得代之的确实是ACC_ SYNCHRONIZED标识,该标识指明了该方法是一个同步方法,JVM通过该ACC. SYNCHRONIZED访问标志来辨别一个方法是否声明为同步方法,从而执行相应的同步调用。
锁升级
synchronized的三种状态,偏向锁,轻量锁,重量锁,不可逆
偏向锁,只有一个线程访问。刚开始的时候线程少,慢慢多起来的,在只有一个线程时使用偏向锁,当线程竞争到个位数时升级到轻量级锁。
轻量级锁,轻量级锁是通过死循环不断去判断有没有资源,没有就继续自旋,并发很多时,自旋消耗很多cpu,导致任务很慢,只适合低并发。自旋次数的默认值是10次,用户可以修改--XX:PreBlockSpin
来更改。
重量级锁,线程都在就绪队列,会对加锁资源进行竞争,未竞争成功的进入阻塞队列(不会产生额外开销),等线程释放竞争资源后,会通知阻塞队列,重新竞争。cpu会全力执行任务。
锁升级原理
在锁对象的对象头里面有一个threadid的字段,
第一次访问的时候,也就是只有一个线程的时候,threadid为空,jvm会让其持有偏向锁,并将threadid设置为线程id。
再次访问时,会先判断threadid与线程id是否一致,如果一致,可以直接使用此对象,如果不一致,这时候也就是有2个线程,偏向锁就会升级为轻量级锁,通过自旋一定次数来获取锁
执行一定次数之后,还没有正常获取使用对象,这时候cpu空转次数多,资源浪费也很高,轻量级锁升级为重量级锁。
这就是锁的升级过程。
**锁升级目的:**锁升级是为了减低锁带来的性能消耗。
作用
1)原子性,是指一个操作要么全部执行要么全部不执行。
2)可见性,可见性是指多个线程访问一个资源时,该资源的状态、值信息等对于其他线程都是可见的。
3)有序性,synchronized保证了每个时刻都只有一个线程访问同步代码块,也就确定了线程执行同步代码块是分先后顺序的,保证了有序性。