Synchronized解析——如果你愿意一层一层剥开我的心

576a2c3096bd0b6afabeab65831484a5.png

若有收获,请记得分享和转发哦

前言

synchronized,是解决并发情况下数据同步访问问题的一把利刃。那么synchronized的底层原理是什么呢?下面我们来一层一层剥开它的心,就像剥洋葱一样,看个究竟。

Synchronized的使用场景

synchronized关键字可以作用于方法或者代码块,最主要有以下几种使用方式,如图:

15db363559f0886728ea8a925eaeb15e.png

接下来,我们先剥开synchronized的第一层,反编译其作用的代码块以及方法

synchronized作用于代码块

public class SynchronizedTest {
    public void doSth(){
        synchronized (SynchronizedTest.class){
            System.out.println("test Synchronized" );
        }
    }
}

反编译,可得:

414b06c3cb97c3a61a69ae4604c94704.png

d3ccc061aa66ad4b9c94dafe746792d2.png

b93850dcb37e0a45fa2e2f354fe53527.png

由图可得,添加了synchronized关键字的方法,多了ACCSYNCHRONIZED标记。即JVM通过在方法访问标识符(flags)中加入ACCSYNCHRONIZED来实现同步功能。

monitorenter、monitorexit、ACC_SYNCHRONIZED

剥完第一层,反编译synchronized的方法以及代码块,我们已经知道synchronized是通过monitorenter、monitorexit、ACC_SYNCHRONIZED实现同步的,它们三作用都是啥呢?我们接着剥第二层:

monitorenter

monitorenter指令介绍

92bbabdab93ea2bf5f85091c6c7bf1b9.png

c12cdb86bbc462e184752a9473aa9658.png

monitorexit

monitorexit指令介绍

389876cc544b38874545ddcbe7bbfb58.png

5dfb91195c7b9cf4e5548f63018aa960.png

ab170ba3beaf5819cbb68def40bdadd2.png

ACC_SYNCHRONIZED

ACC_SYNCHRONIZED介绍

77646c5adf7886b0d02ce34394607d88.png

0056d1ed2c558f5e7bc4a3783d765b4f.png

e5861bd01617b3662618847ea016525c.png

Synchronized第二层的总结

  • 同步代码块是通过monitorenter和monitorexit来实现,当线程执行到monitorenter的时候要先获得monitor锁,才能执行后面的方法。当线程执行到monitorexit的时候则要释放锁。

  • 同步方法是通过中设置ACCSYNCHRONIZED标志来实现,当线程执行有ACCSYNCHRONI标志的方法,需要获得monitor锁。

  • 每个对象维护一个加锁计数器,为0表示可以被其他线程获得锁,不为0时,只有当前锁的线程才能再次获得锁。

  • 同步方法和同步代码块底层都是通过monitor来实现同步的。

  • 每个对象都与一个monitor相关联,线程可以占有或者释放monitor。

好的,剥到这里,我们还有一些不清楚的地方,monitor是什么呢,为什么它可以实现同步呢?对象又是怎样跟monitor关联的呢?客观别急,我们继续剥下一层,请往下看。

054a24e9479b3075b4fbff89953600f5.png

061038d7860b2703f7651c101e14d00f.png

47cf83b7c3786106f4366df6e407c20e.png

ff3a2c0cdfe400c5537c28b571cf2dd4.png

25803694a995e3c001715e61a827177c.png

17e1658b8f1e35476624e08f8864d0ea.png

457f97a0e4335cc189de9b15d730fe94.png

看完上图,其实对象跟monitor怎样关联,我们已经有个大概认识了,接下来我们分对象内存布局,对象头,MarkWord一层层继续往下探讨。

对象的内存布局

在HotSpot虚拟机中,对象在内存中存储的布局可以分为3块区域:对象头(Header),实例数据(Instance Data)和对象填充(Padding)。

5bc18fee926c789528a520538a8bd0cb.png

9ed96b0f27a6527daa5260e4d45a1744.png

1806e806c5246ed5431fbe741c03abcc.png

73247a35c6d4bc4dae86c23f7521c0d6.png

0cd756db6487a99d97d4bba789fa42a4.png

0701b57fb7dcfc8f3ed8d998def0b2ac.png

c8f40ae9c11e1635145f0793ca283242.png

08c724fd8cb3cb09aba5f3252e267cb0.png

总结

我们直接以一张Synchronized洋葱图作为总结吧,如果你愿意一层一层剥开我的心。

5d1fab6c7cf2437cce5a4c0ffc2a6a6c.png

2a57d60f01f503cf23034ef9d0b79204.png

点个在看 就是最好的支持

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值