java互斥锁的实现原理_java-深入分析synchronized原理

互斥锁

互斥锁futex,全拼fast userspace mutexes,直翻为快速用户空间互斥器,它是我们上层应用实现锁的最常用方法。futex是一块所有进程都可以访问的内存,是通过cpu的原子操作修改内存中的值来尝试获取琐,如果没有竞争,则直接在用户空间完成操作,无需切换内核空间,以此保证了futex的性能。

synchronized

其实java多线程操作大体就那么几种,基于cas的aqs,synchronized和volatile,这篇文章主要介绍下synchronized。

synchronized是java的关键字,在老版本的jdk中性能表现并不太好,所以有了很多基于cas的Lock,但最近几版jdk都对synchronized做了很多优化,以后synchronized也会作为jdk主推的锁。

synchronized最主要的优化就是引入了升级功能,升级主要分偏向锁、轻量级锁、重量级锁。

synchronized因为有升级和降级,既不会直接粗暴的使用互斥锁,也不会有cas锁在超高并发下多次尝试引起的性能问题,所以相比juc的cas锁,synchronized在大多数情况下性能更好。

synchronized的使用方法主要有3种

//加锁到静态方法上

public static synchronized void test();

//加锁到实例方法上

public synchronized void test();

//加锁到对象上

synchronized(this){

}

// 加锁在静态方法上同加锁到类对象上

synchronized(Test.class){

}

// 加锁到实例方法上同加锁到当前对象上

synchronized(this){

}

总体上说,不管是锁类对象,还是锁其它对象都是锁到一个对象上了。

fd9007271ccc4a544be2aed33d9de8ef.png

这张图很重要,在下面会多次用到,就先放这里了。

偏向锁

java为了支持锁对象,在对象头上做了上图的设计,markword是对象头中的一部分,64位虚拟机占64bit,markword在不同等级锁状态下存储的内容是不同的,上图是锁处于不同状态时markword存储的内容。

其实经过大量测试在我们使用synchronized时,大多数情况并没有发生竞争,很多访问都发生在一个线程里,所以设计了偏向锁,偏向锁的意思就是锁偏向某个线程。

当我们尝试给一个对象加锁时,会有几种情况。

未偏向:如果lock为01时,biased_lock为0,表示没有线程持有这个对象的偏向锁,线程会通过cas的方式修改对象头获取偏向锁。

可重偏向:如果lock为01时,biased_lock为1,表示对象被偏向锁定,这时还会拿epoch与klass(可以理解为某个Class在虚拟机中对应的对象)的mark_prototype的epoch作比较,如果不一致表示可重偏向,线程会通过cas的方式修改对象头获取偏向锁。

已偏向:lock为01,biased_lock为1,epoch与mark_prototype的epoch相等表示锁已偏向,会比较thread字段的threadId,如果一致表示持有锁的是当前线程,可以重入。

如果没有获取到偏向锁,就需要进行一个撤销偏向锁并升级偏向锁的过程,这个过程比较消耗性能,所以当某类对象,比如我们的User.class这个对象的实例有很多次撤销(默认值为40),虚拟机就会更新klass的epoch,表示可重偏向,这就是为啥锁的是对象,判断epoch是去klass判断。

轻量锁

当未获取到偏向锁时,需要通知持有偏向锁的线程撤销偏向锁,竞争线程则进行一个轻量级锁加锁的过程。

偏向锁的撤销:持有锁的线程进入safepoint时,判断持有锁的线程是否在加锁状态,如果是则直接修改markword为轻量级锁,否则释放偏向锁,表示未锁定、

轻量级锁加锁:虚拟机会在当前线程的栈帧中创建一个lock record,并拷贝markword到lockrecord,再通过cas把对象的markword改为偏向锁,ptr_to_lock_record指向lockrecord,如果cas成功则表示成功获取轻量级锁,否则进行自旋尝试,就是常说的自旋锁。

轻量级锁释放:轻量级锁释放时,只要把lockrecord中的markword替换回对象头的markword就释放成功了。

重量级锁

当自旋尝试次数超过阈值(jvm控制的动态值),锁就会进一步升级,升级为重量级锁。

升级为重量锁后,线程就会出现等待、阻塞、唤醒等各种操作,就会涉及到用户态和内核态的切换,所以叫重量级锁,重量级锁的实现就是文中最开始提到的futex互斥锁实现的。

重量级锁既然需要线程的管理机制,自然引入了管程(monitor),java的管程模式类似mesa。

71b06bfdc688e3e3a1fd6879ef94688d.png

contentionList: 所有想要竞争的线程都要进入的队列,又叫cxq。

entrylist: 准备竞争的线程都在这个队列,这个队列只有空的时候才去cxq中拉去,cxq每次只会有一个进去entrylist。

OnDeck:entrylist中的一个线程,一般为最前面的线程,只有OnDeck线程才会去竞争锁、

waitset:当我们调用wait()方法时,线程就会进入waitset,被notity后直接进入entrylist,所以被唤醒的线程比刚参与竞争的线程优先级更高。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/505548.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

php 自定义加密算法,php自定义加密函数、解密

“/*加密*/function addcoder($str){$yuan abA!c1dB#ef2Cg$h%iD_3jkl^E:m}4n.o{&F*p)5q(G-r[sH]6tuIv7wJxy8z9K0;$jia zAy%0Bx1C$wDv^Eu2-t3(F{sr&G4q_pH5*on6I)m:l7.Jk]j8K}ihgf9#ed!cb[a;if ( strlen($str) 0) return false;for($i 0;$i{for($j 0;$j{if($str[$i…

京东开普勒php接口,IOS菜鸟初学第十五篇:接入京东开普勒sdk,呼起京东app打开任意京东的链接-Go语言中文社区...

我之前写了一篇关于接入京东联盟sdk的文章,但是最近,由于这个原因,如下图导致需要重新集成京东的sdk,但是由于某种原因,因为android和ios端不统一,android接入的是京东开普勒的SDK,这次为了统一…

php用不了for循环吗,php中的这两个for循环有什么区别吗?

如下两个for循环,执行的结果不一样的。for($i 0,$j 0;$i dump($i . - . $j);}echo ;for($i 0;$i dump($i . * . $j);}}结果打印:string(3) "0-0"string(3) "1-1"string(3) "2-2"string(3) "3-3"string(3) &q…

php输出楼层号,ZBlog开发中实现评论楼层号正确输出的具体方法代码

在官方的wiki中,针对评论部分的标签调用是集成了评论楼号的,即标签{$comment.FloorID}。这个标签是不计算子评论的,所以并不是采用key直接计算出的楼号。经过测试,我们会发现,这个楼号存在一定BUG!具体表现…

php 一键登录插件,FastAdmin一键管理插件

FastAdmin中的插件可以通过命令行快速的进行安装、卸载、禁用和启用。准备工作请确保你的FastAdmin已经能正常登录后台请确保php所在的目录已经加入到系统环境变量,否则会提示找不到该命令打开命令行控制台进入到FastAdmin根目录,也就是think文件所在的目…

java代码复数包括虚部和实部,Java中编写Applet程序验证复数类(在问题补充中)实现接收用户输入的复数的实部和虚部,计算复数与复数,复数与实数的加减,乘除操作...

2008-11-16 回答我把方法给你写出来,测试程序的主函数你就自己写把/*** Cigarette ash Inc.* author 烟灰* version 1.00 2008/6/7*///编写一个类ComplexNumber实现复数的运算:public class ComplexNumber {double m_dRealPart;double m_dImaginPart;public Complex…

matlab怎么重新打开新的代码,方程求解程序代码求助-程序代码修改或新的代码...

很简单的方程求解程序,调用mulDNewton函数求解,之前在Matlab 2011b版本上运行成功,现在在Matlab 2018a版本上总是出错,程序代码和出错的提示如下,mulDNewton函数代码也如下。请教大神该程序应该如何修改,请…

matlab实验函数编写与程序设计,matlab实验四函数编写与程序设计.doc

实验四:函数编写与程序设计一、实验目的1 . 掌握M文件的创建。2.掌握函数的编写规则。3.掌握函数的调用。4 . 掌握基本的输入输出函数以及显示函数的用法。5.会用Matlab程序设计实现一些工程算法问题。二、实验内容1 . 设计程序&a…

php静态数组变量初始化,为什么数组初始化时,赋值不能是常量?

刚刚发现的一个很奇怪的现象下面这个数组我设置为类的属性,其中SYSTEM_LIB为我定义的常量final class Application { public static $_libarray( route > SYSTEM_LIB./lib_route.php, mysql > SYSTEM_LIB./lib_mysql.php, );}但最终运行时报错为Parse error: …

php比例算法,图片比例转换算法

项目要求将图片上传的任意图片按固定比例进行转换,没办法自己就写了一个,代码如下:public function cropImage($imagePath, $savePath, $scale){$imgHandler $this->model(Picture_Model_Image);$imgHandler->setSrcImg($imagePath);$…

oracle安装过程掉电,Oracle数据库掉电后ORA-01172磁盘坏块解决方法

由于服务器突然掉电,重启机器后发现数据库无法启动。数据库版本 Oracle10201,OS REDHAT 5.4数据库无归档,无备份 [oraclehuna由于服务器突然掉电,重启机器后发现数据库无法启动。数据库版本 Oracle10201,,O…

php sqlite id自增,Android sqlite设置主键自增长的方法教程

今天在APP中增加一个添加项目的功能,项目的主键为整数,要让它自增长。既然要自增长,那么在代码里面就不用给id字段赋值。但是调试的时候发现不行,提示主键重复,观察了输出,不赋值的话,id默认为0…

oracle flashback 深入研究,oracle 之flashback 深入研究。

oracle 之flashback 深入研究。今天是2013-08-24,开始进行oracle flashback 内部原理研究,记录一下笔记。SQL> startupORACLE instance started.Total System Global Area 405020672 bytesFixed Size 2213816 bytesVariable Size …

linux删除了mount目录,Linux记录-分区(df/fdisk/mount/umount/fuser)

1.查看磁盘挂载(df -TH)2.卸载umount /dev/vdb13.查杀用户进程(fuser -m -v -i -k /dev/vdb1)4.再次卸载umount /dev/vdb1,并查看挂载信息df -TH5.删除分区(fdisk /dev/vdb m d 1 d w)6.查看分区(fdisk -l,没有Start-End磁盘分区表示可用)7.添加分区(fdi…

linux中pak命令,如何在Linux系统中安装Flatpak

选择你的Linux发行版快速安装Flatpak及使用Flatpak。支持Ubuntu、Fedora、Red Hat Enterprise Linux、Deepin、Endless OS、Linux Mint、openSUSE、Arch、Debian、CentOS、Gentoo、Solus、Alpine、Mageia、Pop!_OS、elementary OS、Raspbian。以下为你逐一介绍如何在这些Linux版…

linux服务器组件有哪些,推荐几个linux服务器面板

天兴工作室自zblog从asp换到了php就一直在玩php的程序,php的程序肯定是搭配linux系统更方便一些。现在的云服务器也都不贵几百块钱一年,就算是个人也玩得起。当然linux服务器和windows服务器的用法完全是两样:windows系统就跟本地电脑一样操作…

ubuntu 改linux密码忘了怎么办,Ubuntu 14.04忘记root密码的解决方法

电脑20多天没用,忘记Ubuntu 14.04 root密码了,下面是在网上找到的一个解决办法,其它的和这个也大概相同。因为其中有些缺漏,没能给我解决问题。通过分析最终问题还是解决了,现解决方案的关键点记录一下。希望能方便到其…

drcom linux怎么运行,Drcom_linux

64位ubuntu设置32位环境sudo apt-get install lib32z1 lib32ncurses5 lib32bz2-1.0//目前自己用到sudo apt-get install libsm6:i386sudo apt-get install libxi6:i386sudo apt-get install libxrender1:i386sudo apt-get install libxrandr2:i386sudo apt-get install libxcur…

linux设置历史命令保留数目限制,linux下修改history命令保存条数

在linux系统下、history命令会保存多少条命令呢?曾在一本书上说,如果注销系统,那么会将所有的历史命令都定入到~/.bash_history,但只保留1000条命令(这个是由默认的shell变量所设置了的)但是为什么我们执行history命令后会出现大于1000条的历…

虚拟机linux如何扩大内存吗,如何扩大Vmware虚拟机中Ubuntu系统磁盘空间的方法

在虚拟机里安装系统,有时即会遇到初始分配的磁盘空间不够用的情况。如果是windows系统,则直接在虚拟机选项里扩容即可,但linux系统则没有这么简单。下面是整理的vmware下ubuntu系统磁盘扩容的方法与步骤,仅供参考。提示&#xff1…