Java性能 -- Lock优化
Lock / synchronized
Lock锁的基本操作是通过乐观锁实现的,由于Lock锁也会在阻塞时被挂起,依然属于悲观锁
| synchronized | Lock | |
|---|---|---|
| 实现方式 | JVM层实现 | Java底层代码实现 |
| 锁的获取 | JVM隐式获取 | lock() / tryLock() / tryLock(timeout, unit) / lockInterruptibly() |
| 锁的释放 | JVM隐式释放 | unlock() |
| 锁的类型 | 非公平锁、可重入 | 非公平锁/公平锁、可重入 |
| 锁的状态 | 不可中断 | 可中断 |
| 锁的性能 | 高并发下会升级为重量级锁 | 更稳定 |
实现原理
- Lock锁是基于Java实现的锁,Lock是一个接口
- 常见的实现类:ReentrantLock、ReentrantReadWriteLock,都是依赖AbstractQueuedSynchronizer(AQS)实现
- AQS中包含了一个基于链表实现的等待队列(即CLH队列),用于存储所有阻塞的线程
- AQS中有一个state变量,该变量对ReentrantLock来说表示加锁状态
- AQS中的CLH队列的所有操作均通过CAS操作实现的

锁分离优化
ReentrantReadWriteLock
- ReentrantLock是一个独占锁,同一时间只允许一个线程访问
- ReentrantReadWriteLock允许多个读线程同时访问,但不允许写线程和读线程、写线程和写线程同时访问
- ReentrantReadWriteLock内部维护了两把锁,一把用于读操作的ReadLock,一把用于写操作的WriteLock
- ReentrantReadWriteLock如何保证共享资源的原子性?ReentrantReadWriteLock也是基于AQS实现的
- 自定义同步器(继承AQS)需要在同步状态state上维护多个读线程和一个写线程的状态
- ReentrantReadWriteLock利用了高低位,来实现一个整型控制两种状态的功能
- 将同步状态state切分为两部分,高16位表示读,低16位表示写
获取写锁
- 一个线程尝试获取写锁时,会先判断同步状态state是否为0
- 如果state为0,说明暂时没有其他线程获取锁
- 如果state不为0,说明其它线程获取了锁
- 当state不为0时,会再去判断同步状态state的低16位(w)是否为0
- 如果w为0,说明其它线程获取了读锁,此时直接进入CLH队列进行阻塞等待(因为读锁与写锁互斥)
- 如果w不为0,说明有线程获取了写锁,此时要判断是不是当前线程获取了写锁
- 如果不是,进入CLH队列进行阻塞等待
- 如果是,就应该判断当前线程获取写锁是否超过最大次数,如果超过,抛出异常,否则更新同步状态state

获取读锁
- 一个线程尝试获取读锁时,同样会先判断同步状态state是否为0
- 如果state为0,说明暂时没有其他线程获取锁,此时需要判断是否需要阻塞
- 如果需要阻塞,则进入CLH队列进行阻塞等待
- 如果不需要阻塞,则CAS更新state为读状态
- 如果state不为0,说明其它线程获取了锁
- 如果state为0,说明暂时没有其他线程获取锁,此时需要判断是否需要阻塞
- 当state不为0时,会同步判断同步状态state的低16位
- 如果存在写锁,直接进入CLH阻塞队列
- 反之,判断当前线程是否应该被阻塞,如果不应该被阻塞则尝试CAS同步状态,获取成功更新同步锁为读状态

StampedLock
- ReentrantReadWriteLock被很好地应用在读多写少的并发场景中,但会存在写线程饥饿的问题
- Java 8引入StampedLock解决了这个问题
- StampedLock不是基于AQS实现的,但实现原理与AQS类似,都是基于队列和锁状态
- StampedLock有三种模式:写、悲观读、乐观读,StampedLock在获取锁时会返回一个票据stamp
- 一个写线程获取写锁的过程中,首先是通过writeLock获取一个票据stamp(表示锁的版本)
- WriteLock是一个独占锁,同时只能有一个线程可以获取WriteLock
- 当一个线程获取WriteLock后,其他请求的线程必须等待
- 当没有其他线程持有读锁或者写锁时才可以获得WriteLock
- 一个读线程获取读锁的过程中,首先会通过tryOptimisticRead获取一个票据stamp
- 如果当前没有线程持有写锁,会返回一个非0的stamp
- 然后调用validate验证之前调用tryOptimisticRead返回的stamp在当前是否有其他线程持有了写锁
- 如果是,那么validate返回0,升级为悲观锁
- 相对于ReentrantReadWriteLock,StampedLock获取读锁只使用了与或操作进行校验,不涉及CAS操作
- 即使第一次乐观锁获取失败,也会马上升级为悲观锁,可以避免一直进行CAS操作而带来的CPU性能消耗问题
- 但StampedLock并没有被广泛使用,有几个主要原因
- StampedLock的功能仅仅只是ReadWriteLock的子集
- StampedLock不支持重入!!
- StampedLock的悲观读锁、写锁都不支持条件变量(不符合管程模型)
1 |
public class Point {
|
小结
- 不管使用synchronized同步锁还是Lock同步锁,只要存在锁竞争就会产生线程阻塞,导致线程频繁切换,增加性能消耗
- 优化锁的关键:降低锁竞争
- synchronized同步锁:减少锁粒度、减少锁占用时间
- Lock同步锁:锁分离
最后,我是小架
我们下篇文章见!
Java性能 -- Lock优化的更多相关文章
- Java性能之优化RPC网络通信
服务框架的核心 大型服务框架的核心:RPC通信 微服务的核心是远程通信和服务治理 远程通信提供了服务之间通信的桥梁,服务治理提供了服务的后勤保障 服务的拆分增加了通信的成本,因此远程通信很容易成为系统 ...
- 推荐:Java性能优化系列集锦
Java性能问题一直困扰着广大程序员,由于平台复杂性,要定位问题,找出其根源确实很难.随着10多年Java平台的改进以及新出现的多核多处理器,Java软件的性能和扩展性已经今非昔比了.现代JVM持续演 ...
- Java基础学习总结(80)——Java性能优化详解
让Java应用程序运行是一回事,但让他们跑得快就是另外一回事了.在面对对象的环境中,性能问题就像来势凶猛的野兽.但JVM的复杂性将性能调整的复杂程度增加了一个级别.这里Refcard涵盖了JVM in ...
- Java 性能优化之 String 篇
原文:http://www.ibm.com/developerworks/cn/java/j-lo-optmizestring/ Java 性能优化之 String 篇 String 方法用于文本分析 ...
- java 性能优化(代码优化)
参考博文: java 性能优化:35 个小细节,让你提升 java 代码的运行效率
- java 性能优化:35 个小细节,让你提升 java 代码的运行效率
前言 代码 优化 ,一个很重要的课题.可能有些人觉得没用,一些细小的地方有什么好修改的,改与不改对于代码的运行效率有什么影响呢?这个问题我是这么考虑的,就像大海里面的鲸鱼一样,它吃一条小虾米有用吗?没 ...
- 读书笔记系列之java性能优化权威指南 一 第一章
主题:java性能优化权威指南 pdf 版本:英文版 Java Performance Tuning 忽略:(0~24页)Performance+Acknowledge 1.Strategies, A ...
- [原创]Java性能优化权威指南读书思维导图
[原创]Java性能优化权威指南读书思维导图 书名:Java性能优化权威指南 原书名:Java performance 作者: (美)Charlie Hunt Binu John 译者: 柳飞 ...
- 如何优化 Java 性能?
对于 Java 性能比较关心的同学大概都知道<Java Performance>这本书,一般而言,很多同学在日常写 Java Code 的时候很少去关心性能问题,但是在我们写 Code 的 ...
随机推荐
- linux命令-挂载命令
一.挂载命令 1.mount 命令基本格式 linux 所有存储设备都必须挂载使用,包括硬盘 命令名称:mount 命令所在路径:/bin/mount 执行权限:所有用户 [root@localhos ...
- 【Hash一致性算法】什么是Hash一致性算法
目录 1. 一致性Hash算法简介 环形Hash空间 把数据通过一定的hash算法处理后映射到环上 将机器通过hash算法映射到环上 机器的删除与添加 平衡性 本文转载自博客 1. 一致性Hash算法 ...
- java遍历request.getParameterMap()中的值
在开发过程中发现request对象有提供一个request.getParameterMap()方法可以获取到从前端请求发送的参数Map. 但是在使用get()方法通过key(键)去获取这个参数Map中 ...
- 一年半前端工作经验试水杭州:我是如何拿下网易、阿里和滴滴 offer 的
前言 笔者毕业于东北大学,大学毕业社招进入环球网,前端开发工程师一职.技术栈:React+node,Github 地址 成果 来到杭州的目标非常的明确,大厂.其实就是网易.阿里和滴滴.好在基本三家都拿 ...
- 一起学Android之ContentProvider
本文主要讲解在Android开发中ContentProvider的常规用法,仅供学习分享使用,如有不足之处,还请指正. 访问一个ContentProvider 在Android开发中,应用程序通过Co ...
- Mybatis1
一.MyBatis介绍 本是apache的一个开源项目iBatis, 2010年这个项目由apache software foundation 迁移到了google code,并且改名为MyBatis ...
- Java分布式系统---消息中间件
简介 消息中间件也可以称消息队列,是指用高效可靠的消息传递机制进行与平台无关的数据交流,并基于数据通信来进行分布式系统的集成.通过提供消息传递和消息队列模型,可以在分布式环境下扩展进程的通信.当下主流 ...
- 服务器返回的数据将Unicode码转成汉字
当我们请求接口的时候,服务器会返回一些数据,当我们打印的时候就会发现,打印出来的是unicode码,不是汉字. 这时候需要我们自己手动处理一下,让打印的时候输出汉字的格式. 方法如下: 新增一个分类, ...
- Jupyter Notebooks 配置
重装了三遍(破音) 一.首先进行Anaconda的下载 然后安装,将环境配置到系统变量上,如下 然后,打开 windows 的终端,检查是否配置成功 conda -V 然后就可以开始 Jupyter ...
- Good start is a half success(2019-04-07)
一. 回顾你过去将近3年的学习经历. (1)当初你报考的时候,是真正喜欢计算机这个专业吗?. (2)你现在后悔选择了这个专业吗?. (3)你认为你现在最喜欢的领域是什么(可以是计算机的也可以是其它领域 ...