Java中synchronized关键字引出的多种锁 问题
作者:天降e包only 发布时间:2021-06-05 04:42:45
前言
Java 中的 synchronized
关键字可以在多线程环境下用来作为线程安全的同步锁。本文不讨论 synchronized
的具体使用,而是研究下synchronized
底层的锁机制,以及这些锁分别的优缺点。
一 synchronized机制
synchronized
关键字是JAVA中常用的同步功能,提供了简单易用的锁功能。synchronized
有三种用法,分别为:
用在普通方法上,能够锁住当前对象。用在静态方法上,能够锁住类用在代码块上,锁住的是synchronized
()里的对象
在JDK6之前,synchronized
使用的是重量级锁制,在之后synchronized
加入了锁膨胀机制,显著提升了synchronized
关键字的效率。
基于synchronized
关键字,我们来了解下几种类别的锁,并且讲解synchronized
的锁膨胀机制。
synchronized
锁是非公平锁。并且一个被synchronized
锁住的对象或类,就是一把锁。
另外一提,所有锁都是存储在Java对象头里的,Java对象头里的Mark Word里默认存储对象的HashCode,分代年龄和锁标记位。也就是说Mark Word记录了锁的状态
二 锁膨胀机制与几类锁
锁膨胀是不可逆的
2.1 偏向锁
synchronized
在JDK1.6以后默认开启偏向锁
,synchronized
最初都是偏向锁
表现:一个线程获取锁成功后,会在对象头里记录线程ID,以后该线程获取和释放锁都没有任何花费。(因为该锁已经被绑定在该线程上了,且在膨胀前不会改变),如果其他线程尝试获取这个锁,偏向锁
将会膨胀为轻量锁
。
优点:在只有一个线程使用锁的时候获取和退出锁没有任何花费
缺点:锁竞争激烈会很快升级为轻量锁
,那么维持偏向锁
的过程就是在浪费计算机资源。(不过因为偏向锁
本身就很轻量,因此浪费的资源并不多)
小结:只有一个线程使用锁的情况下,synchronized
使用的锁为偏向锁
。
如果锁竞争激烈,可以通过配置JDK禁用偏向锁
。
2.2 轻量锁
一把锁不止一个线程使用,则偏向锁
膨胀为轻量锁
表现:线程获取轻量锁
时,会直接用CAS
修改对象头里锁的记录,如果修改失败,代表此时锁存在多个线程的竞争,轻量锁
将会膨胀为重量锁
。
优点:在线程之间使用锁不存在竞争时,一次CAS
操作就能获取和退出锁
缺点:与偏向锁
类似
小结:只要一把锁不止一个线程获取过,偏向锁
就会膨胀为轻量锁
。
2.3 重量锁
一把锁存在多线程竞争,则轻量锁
开始自旋,自旋一定次数后仍没获取锁,则膨胀为重量锁
(存在竞争时,轻量锁
虽然会先自旋,但是最终往往都会膨胀为重量锁
)
表现:线程获取重量锁
时,如果获取失败(即锁已被其他线程获取),则使用自适应自旋锁
,自旋一定次数后仍没获取锁,则进入阻塞队列等待。
优点:未获取到的锁进入阻塞队列,节约CPU资源。(好吧感觉其实是没有啥优点)
缺点:重量锁
是通过对象内部的监视器(monitor)实现,其中monitor的本质是依赖于底层操作系统的Mutex Lock实现,操作系统实现线程之间的切换需要从用户态到内核态的切换,切换成本非常高。
小结:只要一把锁存在多线程竞争,轻量锁
就会膨胀为重量锁
。
自旋锁
synchronized
的轻量锁
,重量锁
,使用了自适应自旋锁
进行性能优化
首先介绍自旋锁
表现:线程获取锁失败后,不会进入阻塞等待,而是再次尝试去获取锁,如此反复,直到获取到锁,或者自旋结束那么会阻塞等待。
解决问题:在某些场景下,线程持有锁的时间非常短。在线程获取锁失败后,如果线程进入阻塞将会带来线程上下文的切换,上下文切换的时间可能反而高于线程反复尝试获取锁的时间。
此时线程原地等待去重复获取锁。反而在性能上更有优势。
缺点:
单核CPU没有线程并行,反复尝试会导致进程无法继续运行。重复尝试导致了CPU的占用,如果CPU资源紧张的话反而会性能下降如果锁的竞争时间过长,不仅没有性能提升,还浪费了大量CPU资源。
优化:使用自适应自旋锁
。自适应自旋锁会根据之前的锁获取记录,优化调整自旋时间,避免造成不必要的自旋。
三 具体synchronized流程
总结
以上所述是小编给大家介绍的Java中synchronized关键字引出的多种锁 问题 ,网站的支持!
如果你觉得本文对你有帮助,欢迎转载,烦请注明出处,谢谢!
来源:https://www.cnblogs.com/taojinxuan/p/11139398.html


猜你喜欢
- 这个工具叫“InstallShield”,可以自己去网上下一个,有绿色版本 也有安装版的。 &
- 在 String 中提供了两个截取字符串的方法,一个是从指定位置截取到字符串结尾,另一个是截取指定范围的内容。下面对这两种方法分别进行介绍。
- 前言在某些使用了readonly关键字的情况下,C#编译器会创建出结构体的防御副本。虽然这个问题已经众所周知并被记录下来了,但仍然值得重新审
- 何时需要削峰当上游调用下游服务速率高于下游服务接口QPS时,那么如果不对调用速率进行控制,那么会发生很多失败请求通过消息队列的削峰方法有两种
- 背景以springboot为tomcat启动的框架,以angular2为前端页面的框架,最后需要将angular2的代码运行在springb
- 在实际项目使用中,必须要考虑服务的安全性,当服务部署到互联网以后,就要考虑服务被恶意请求和暴力攻击的情况,下面的教程,通过intercept
- 只要是面向对象的编程语言,基本上都有类Class的用法,只是好不好用,好不好记而已,面向对象是c++开始引入的,但是c++ 关于类的东西,弄
- Java是垃圾回收语言的一种,其优点是开发者无需特意管理内存分配,降低了应用由于局部故障(segmentation fault)导致崩溃,同
- 我们需要获取Android手机或Pad的屏幕的物理尺寸,以便于界面的设计或是其他功能的实现。下面就分享一下Android中常用的一些辅助方法
- 前言最近项目targetSdkVersion升级到了26,出现很多问题趟了很多坑,其中就包括本篇的需要解决的问题:全局dialog 不显示。
- 自动装箱和拆箱问题是Java中一个老生常谈的问题了,今天我们就来一些看一下装箱和拆箱中的若干问题。本文先讲述装箱和拆箱最基本的东西,再来看一
- TimeLineStepView支持时间轴和StepView,三种布局,支持水平布局,垂直布局和自定义布局,截图如下
- 1.概述项目中经常会遇到一个应用需要访问多个数据源的情况,本文介绍在SpringBoot项目中利用SpringDataJpa技术如何支持多个
- 一. ANR场景无论是四大组件或者进程等只要发生ANR,最终都会调用AMS.appNotResponding()方法,下面从这个方法说起。以
- List list=new ArrayList()是怎么回事首先明确List是接口,ArrayList是它的实现类以下两种方法都可以,但是不
- 实践过程效果代码public partial class Form1 : Form{ public Form1()
- 三种定义数组的格式如下:int[] arr1=new int[10];int[] arr2={1,2,3,6};int[] arr3=new
- Java提供一种机制叫做序列化,通过有序的格式或者字节序列持久化java对象,其中包含对象的数据,还有对象的类型,和保存在对象中
- 双向链表(Doubly linked list)什么是双向链表?双向链表也叫双链表,是链表的一种,它的每个数据结点中都有两个指针,分别指向直
- java弱口令检测机制1. 设计要求应具备检测口令的长度和是否在指定字符集合内的能力。应具备检测口令字符逻辑相邻的能力,如aBc,abC等。