[学习笔记]java虚拟机(三)

书接上文,虚拟机(二)

锁在虚拟机的实现优化

基本概念

java的对象中都有一个对象头,用于保存对象的系统信息。对象头中有一个成为Mark Word的部分,是一个数据区,存放对象的哈希值、对象的年龄、锁的指针等信息。


偏向锁

某个锁被线程获取后,就会进入偏向模式,线程再次请求这个锁时,无需再进行相关的同步操作。

偏向锁在竞争不激烈的系统中能够很好地提升性能,但激烈的场合没有太强的优化效果。

  • -XX:+UseBiasedLocking设置启用偏向锁。
  • -XX:-UseBiasedLocking设置禁用偏向锁。
  • -XX:+BiasedLockingStartupDelay:设置虚拟机启动后立即启用偏向锁。虚拟机默认启动后4秒后,才启用偏向锁。

轻量级锁

偏向锁失败后,虚拟机会让线程申请轻量级锁。轻量级锁在虚拟机内部的实现是一个BasicObjectLock的对象,对象内部包括BasicLock对象和一个持有该锁的java对象指针。BasicLock对象内部维护的displaced_header字段备份了对象头部的Mark Word。整个BasicObjectLock对象放在java栈的栈针中。

线程持有一个对象的锁时,对象头部的Mark Word为指向BasicObject对象的指针。因为BasicObjectLock对象在栈中,指针必然指向线程栈空间。当需要判断某线程是否持有该对象锁时,只需简单地判断对象头的指针是否在当前线程的栈地址范围内。同时,BasicLock对象的displaced_header字段备份了原对象的Mark Word,BasicObjectLock对象的obj字段指向该对象。

BasicLock备份后,使用CAS操作尝试将BasicLock的地址复制到对象头的Mark Word。复制成功则加锁成功,否则可能会膨胀为重量级锁。


锁膨胀

轻量级锁膨胀为重量级锁时,首先废弃前面BasicLock备份的对象头信息。然后正式启用重量级锁。
启用过程:通过inflate()方法进行锁膨胀,获得对象的ObjectMonitor;然后使用enter()方法尝试进入该锁。调用enter方法可能会让线程在操作系统层面被挂起。


自旋锁

为避免线程在操作系统层面被挂起。虚拟机会采取自旋锁的方法。

自旋锁会让线程在未取得锁时,去执行空循环(自旋),若干个空循环后,若仍不能获得锁,才会被挂起。

在锁竞争不激烈、锁占用时间短的系统中,具有优化性能。

-XX:+UseSpinning:开启自旋锁。
-XX:PreBlockSpin:设置自旋锁的等待次数。
1.7版本的java之后,虚拟机不支持用户配置自旋锁。


锁消除

java虚拟机在JIT编译时,通过对运行上下文的扫描,去除不可能存在共享资源竞争的锁。

  • -XX:+DoEscapeAnalysis:逃逸分析。
  • -XX:+EliminateLocks:开启锁消除。(必须工作在-server模式下)

锁在应用层的优化

减少锁持有时间

在必要时同步,局部加锁。


减少锁粒度

该优化方法的典型应用是ConcurrentHashMap类的实现。

拆分锁对象:缩小锁定对象的范围,从而减小锁冲突的可能性。会引起的问题是系统需要获取全局锁时,消耗的资源的会比较多。如ConcurrentHashMap的size方法。


锁分离

将独占锁分成多个锁。典型应用是LinkedBlockingQueue:put和take操作并不冲突,用两把不同的锁分离了两个操作。


锁粗化

虚拟机在遇到一连串连续的对同一锁不断进行请求和释放的操作时,便会把所有的锁操作整合成对锁的以此请求,减少对锁的请求同步次数。
例如循环中的锁。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值