如何看到synchronized背后的monitor锁?
本文我们研究下 synchronized 背后的 monitor 锁。获取和释放 monitor 锁的时机
我们都知道,最简单的同步方式就是利用 synchronized 关键字来修饰代码块或者修饰一个方法,那么这部分被保护的代码,在同一时刻就最多只有一个线程可以运行,而 synchronized 的背后正是利用 monitor 锁实现的。所以首先我们来看下获取和释放 monitor 锁的时机,每个 Java 对象都可以用作一个实现同步的锁,这个锁也被称为内置锁或 monitor 锁,获得 monitor 锁的唯一途径就是进入由这个锁保护的同步代码块或同步方法,线程在进入被 synchronized 保护的代码块之前,会自动获取锁,并且无论是正常路径退出,还是通过抛出异常退出,在退出的时候都会自动释放锁。
我们首先来看一个 synchronized 修饰方法的代码的例子:public synchronized void method() { method body } 复制代码
我们看到 method() 方法是被 synchronized 修饰的,为了方便理解其背后的原理,我们把上面这段代码改写为下面这种等价形式的伪代码。public void method() { this.intrinsicLock.lock(); try{ method body } finally { this.intrinsicLock.unlock(); } } 复制代码
在这种写法中,进入 method 方法后,立刻添加内置锁,并且用 try 代码块把方法保护起来,最后用 finally 释放这把锁,这里的 intrinsicLock 就是 monitor 锁。经过这样的伪代码展开之后,相信你对 synchronized 的理解就更加清晰了。用 javap 命令查看反汇编的结果
JVM 实现 synchronized 方法和 synchronized 代码块的细节是不一样的,下面我们就分别来看一下两者的实现。同步代码块
首先我们来看下同步代码块的实现,如代码所示。public class SynTest { public void synBlock() { synchronized (this) { System.out.println("lagou"); } } } 复制代码
在 SynTest 类中的 synBlock 方法,包含一个同步代码块,synchronized 代码块中有一行代码打印了 lagou 字符串,下面我们来通过命令看下 synchronized 关键字到底做了什么事情:首先用 cd 命令切换到 SynTest.java 类所在的路径,然后执行 javac SynTest.java,于是就会产生一个名为 SynTest.class 的字节码文件,然后我们执行 javap -verbose SynTest.class,就可以看到对应的反汇编内容。
关键信息如下: public void synBlock(); descriptor: ()V flags: ACC_PUBLIC Code: stack=2, locals=3, args_size=1 0: aload_0 1: dup 2: astore_1 3: monitorenter 4: getstatic #2 // Field java/lang/System.out:Ljava/io/PrintStream; 7: ldc #3 // String lagou 9: invokevirtual #4 // Method java/io/PrintStream.println:(Ljava/lang/String;)V 12: aload_1 13: monitorexit 14: goto 22 17: astore_2 18: aload_1 19: monitorexit 20: aload_2 21: athrow 22: return 复制代码
从里面可以看出,synchronized 代码块实际上多了 monitorenter 和 monitorexit 指令,标红的第3、13、19行指令分别对应的是 monitorenter 和 monitorexit。这里有一个 monitorenter,却有两个 monitorexit 指令的原因是,JVM 要保证每个 monitorenter 必须有与之对应的 monitorexit,monitorenter 指令被插入到同步代码块的开始位置,而 monitorexit 需要插入到方法正常结束处和异常处两个地方,这样就可以保证抛异常的情况下也能释放锁
可以把执行 monitorenter 理解为加锁,执行 monitorexit 理解为释放锁,每个对象维护着一个记录着被锁次数的计数器。未被锁定的对象的该计数器为 0,我们来具体看一下 monitorenter 和 monitorexit 的含义:monitorenter
执行 monitorenter 的线程尝试获得 monitor 的所有权,会发生以下这三种情况之一:
a. 如果该 monitor 的计数为 0,则线程获得该 monitor 并将其计数设置为 1。然后,该线程就是这个 monitor 的所有者。
b. 如果线程已经拥有了这个 monitor ,则它将重新进入,并且累加计数。
c. 如果其他线程已经拥有了这个 monitor,那个这个线程就会被阻塞,直到这个 monitor 的计数变成为 0,代表这个 monitor 已经被释放了,于是当前这个线程就会再次尝试获取这个 monitor。monitorexit monitorexit 的作用是将 monitor 的计数器减 1,直到减为 0 为止。代表这个 monitor 已经被释放了,已经没有任何线程拥有它了,也就代表着解锁,所以,其他正在等待这个 monitor 的线程,此时便可以再次尝试获取这个 monitor 的所有权。同步方法
从上面可以看出,同步代码块是使用 monitorenter 和 monitorexit 指令实现的。而对于 synchronized 方法,并不是依靠 monitorenter 和 monitorexit 指令实现的,被 javap 反汇编后可以看到,synchronized 方法和普通方法大部分是一样的,不同在于,这个方法会有一个叫作 ACC_SYNCHRONIZED 的 flag 修饰符,来表明它是同步方法。
同步方法的代码如下所示。public synchronized void synMethod() { } 复制代码
对应的反汇编指令如下所示。 public synchronized void synMethod(); descriptor: ()V flags: ACC_PUBLIC, ACC_SYNCHRONIZED Code: stack=0, locals=1, args_size=1 0: return LineNumberTable: line 16: 0 复制代码
可以看出,被 synchronized 修饰的方法会有一个 ACC_SYNCHRONIZED 标志。当某个线程要访问某个方法的时候,会首先检查方法是否有 ACC_SYNCHRONIZED 标志,如果有则需要先获得 monitor 锁,然后才能开始执行方法,方法执行之后再释放 monitor 锁。其他方面, synchronized 方法和刚才的 synchronized 代码块是很类似的,例如这时如果其他线程来请求执行方法,也会因为无法获得 monitor 锁而被阻塞。
好了,本文的内容就全部讲完了,我们讲解了获取和释放 monitor 的时机,以及被 synchronized 修饰的等价代码,然后我们还利用 javac 和 javap 命令查看了 synchronized 代码块以及 synchronized 方法所对应的的反汇编指令,其中同步代码块是利用 monitorenter 和 monitorexit 指令实现的,而同步方法则是利用 flags 实现的。
原文链接:https://juejin.cn/post/7140204404908490788
九十年代大部分人月收入三五百元,为什么很多人舍得买一两千衣服?人多什么人都有,有喜欢吃喝的,有喜欢打扮的,有的家底厚,有的孩子多家里穷,是根据各自嗜好按排的,有的买两三千元上档的衣服,几年几十年不用再买,买一件担当一件,也不见得年年买,多少件
哪种理财方式最安全最适合老百姓购买?最安全最适合老百姓购买的我认为有三种产品国债定期存款及货币基金。风险可以认为基本没有,收益率比活期存款高出6倍到10倍左右。第一,国债。国债产品我认为最适合不愿意打理理财的人选择,
小腿粗的女生要穿怎样的衣服能显得腿细一些呢?想要显腿细?当然是减肥啊,没有什么是比减肥更见效的了,如果是小腿粗,就一定要加强拉伸和按摩言归正传,假如想通过穿衣服显腿瘦,也是一个非常不错的办法,而且很容易学会,同时还能打扮自己
短款风衣怎么搭配?风衣可是春天非常必要的单品啊!而今年的时尚圈,好像对超长款式的风衣已经不是那么感兴趣了,反而短款的风衣占据了时尚的话题榜!那短款风衣要怎么搭配才好看呢?首先我非常喜欢收腰款式的风衣
为什么有些人很有钱,却总是穿很便宜的衣服?这与个人性格和生活习惯有关,跟有没有钱没多大关系。有些人有钱,但很低调,有些人没钱,确要打肿脸充胖子。有些人有钱,满世界招摇,生怕别人不知道,有些人没钱,勤俭持家,小日子照样过得红
如何处理既舍不得扔掉,利用率又低的衣服?女人的衣服随着四季更替,换季怎样快速整理衣服,让你的衣柜井井有条,这里总结了整理的6个步骤,每一个女人都能学会。Step1清空衣柜打开网易新闻查看更多精彩图片将衣柜里所有的衣服仍在
退休后有什么增加收入的方式?退休后增加收入的方法有很多,但我觉得退休后以节约为主,减少开支,就是挣钱。不能去干体力活了,万一受伤了,就得不偿失了。总结起来有以下几条一,打理好手中的钱,这是最根本的收入和本钱。
当工作与理想有差距,你会选择妥协还是放弃?感谢头条的邀请,很高兴回答这个问题。我们从小就被教育要有一个远大的理想,并且要认真学习为之而努力奋斗。但当我们走出校园去寻找心中的那片伊甸园的时候,往往会发现现实与自己的理想是有一
莆田有哪些适合带女朋友去的地方?莆田是个好地方呢,小编给你介绍一下,希望可以帮到你,也祝你们两个玩的开心呀湄洲岛湄洲岛位于福建省莆田市中心东南42公里,距大陆仅1。82海里,是莆田市第二大岛,陆域面积14。35平
请问买房的时候有公积金,等月供2年后换了工作没有公积金,贷款会不会变成商贷了呢?谢谢邀请。前两天作者正好遇到过一个类似的问题,具体的情况让我来跟大家详细说明一下,毕竟不是所有人都明白流程!应该是在本周一的时候,作者二姑家的妹妹打电话过来,问公积金贷款的事情。她
有没有住在惠州的朋友,给提供一下惠州比较靠谱的楼盘和房价?我在惠州生活了二十四年,亲身经历了楼市的变迁。我不是中介,爱好看楼,专业看楼二十年。惠州很大,目前比较热门的片区是惠城金山湖惠城北站新城临深大亚湾西区,临深惠阳白云新区。惠城的仲恺
最近想换个手机,1000到2000元左右,续航好颜值高充电快的手机有哪些?题主想要10002000元之间的长续航和拍照好的手机,这个价位买到的就是安卓中端手机,甚至会偏低端一些的手机。小编今天就给大家推荐几款这样的手机。第一款红米K30在我看来,最符合题
要想上警校的话需要高考考多少分?警校分为两种,一种是提前批的,分数需要达到一本分数线以上,不过也有二本的院校,但是分数绝对不会低。另一种是普通批次的警校,一般都是高职高专。这两种警校的有什么区别呢?提前批。提前批
尿酸在420480之间长达6年,有什么危害?近年来,痛风的患病率呈明显的逐年上升的趋势,并且还存在时间地域种族年龄和性别上的差异。既往认为长期持续的尿酸居高不下,尤其是45岁以上的男性容易患痛风现在对于痛风的认识已经发生变化
一个有500亿现金,一个市值有5000亿,到底哪个有钱?你这个问题有点考验人性。一个是现金500亿,一个是市值5000亿,到底哪一个更有钱?你问的是这二者到底哪一个更有钱,从这个角度看,毫无疑问,当然是拥有5000亿市值的人更有钱。为什
右上腹疼痛是什么原因?如何应对?在我看诊过的患者中,无论是急性还是慢性肝炎,或者是肝硬化和肝癌,都经常会有右上腹痛的情况。他们的右上腹痛,有时并不单单是肝脏的原因,还可能是胆,或者十二指肠食道肠道,甚至是由肺胸膜
现在中考越来越难,每年没有考上高中的学生都是什么原因造成的?再过两个多月一年一度的中考就要到来了,每年中考过后都会有很多初中生没能考入高中,那么都是因为什么原因造成这些孩子没有考上高中呢?接下来就这个问题进行详细的阐述,同时也在这里告诫每一
孩子写作业磨蹭,是什么原因?小孩写作业很慢的原因有三一是写字的熟练程度,影响写作业的速度二是小孩对知识的掌握程度,需思考答案三是小孩的性格,习惯,可能写作业时还做其他事情。要想提高写作业的速度,应从这三方面入
靶向药,可以停药吗?作为一乡镇卫生院的小医生我就和你讲讲我的小见闻吧,我收了一个老年肺癌患者本来他一直在吃了半年靶向药物,复查ct癌块无明显生长,然后老爷子嫌弃药物副作用大吃了浑身无力,自行停药半个月
网上被诈骗报案后成功取回的几率有多大?之前看了资料的,大部分诈骗犯在境外,很多也是被境外组织控制犯罪的。很多人被骗过去,然后在那些非法组织的犯罪团伙控制下,按照既定的剧本寻找目标。比如杀猪盘,他们往往是很多人去分析受害
马桶买什么好?国内的我建议买恒洁和惠达(唐山),恒洁节水,惠达性价比较高,冲得也很干净!我买的就是惠达。如果还要买得更好点的话,当然是TOTO和科勒了,这两个很有特色。TOTO很厉害了,有一项虹
农村有一种叫刺蓟草,它有哪些功效和作用呢?农村有一种叫刺薊草,它有哪些功效和作用呢?说起刺薊草相信会有很多人不太知道它是什么,但要叫它小名刺菜或是七七菜止血菜,这在农村是无人不知无人不晓得。刺菜是农村最常见的一种野菜,在我