synchorized锁升级过程:
synchorized锁升级过程中只能升级不能降级,起初是JDK早期(1.5之前),是重量级锁,是找操作系统申请OS锁。所谓重量级锁是说获取锁和释放锁都需要经过操作系统,需要经过相当多的步骤,一旦线程或许锁失败,整个系统都会陷入阻塞状态,风险等级很高,虽说synchorized出现异常会自动释放,但是释放过程会牵涉到用户态到核心态之间切换是个性能开销非常大的操作。
而在JDK1.6之后有了一个无锁-->偏向锁-->自旋锁-->重量级锁。过程大致内容是:在线程来了之后,先不尝试对线程进行加锁,只是记录线程的id值,就默认为这个对象是这个线程独有;如果线程锁中已经有线程id记录,则后来的线程就会先进行自旋,默认自旋10次,如果依旧拿不着锁,就会升级为重量级锁;升级为重量级锁之后,就会进入waitting区,不再占用cpu。
锁的底层标志
在JVM中,普通对象在内存中分为三块区域:对象头,实例数据和对齐填充数据。而对象头包括markword和类型指针,实例数据就是对象的成员变量,padding就是为了保证对象的大小为8字节的倍数,将对象所占字节数补到能被8整除。
其中,一个普通对象有16个字节,markwork占8个字节,类型指针会分情况有不同数量字节(开启压缩指针4字节,不开启8字节,如果是32g以上内存,都是8字节)。
而加锁其实就是在对象的markwork前两位二进制码设个标记,具体为:00(无锁),01(偏向锁),10(自旋锁),11(OS锁)
提醒
synchorized锁方法时,锁的是this,整个方法;
锁静态时,锁的是.class,整个class文件;
其中锁定方法和非锁定方法可以同时进行。