在后面的文章《青铜4:synchronized用法初体验》中,咱们曾经提到的概念,并指出synchronized是锁机制的一种实现。可是,这么说未免太过形象,你可能无奈直观地了解锁到底是什么?所以,本文会粗略地介绍synchronized背地的一些基本原理,让你对Java中的锁有个粗略但直观的印象。

本文将分两个局部,首先你要从Mark Word中意识锁,因为对象锁的信息存在于Mark Word中,其次通过JOL工具理论体验Mark Word的变动。

一、从Mark Word意识锁

咱们晓得,在HotSpot虚拟机中,一个对象的存储散布由3个局部组成:

  • 对象头(Header):由Mark WordKlass Pointer组成;
  • 实例数据(Instance Data):对象的成员变量及数据;
  • 对齐填充(Padding):对齐填充的字节,临时不用理睬。

在这3个局部中,对象头中的Mark Word是本文的重点,也是了解Java锁的要害。Mark Word记录的是对象运行时的数据,其中包含:

  • 哈希码(identity_hashcode)
  • GC分代年龄(age)
  • 锁状态标记
  • 线程持有的锁
  • 偏差线程ID(thread)

所以,从对象头中的Mark Word看,Java中的锁就是对象头中的一种数据。在JVM中,每个对象都有这样的锁,并且用于多线程拜访对象时的并发管制。

如果一个线程想拜访某个对象的实例,那么这个线程必须领有该对象的。首先,它须要通过对象头中的Mark Word判断该对象的实例是否曾经被线程锁定。如果没有锁定,那么线程会在Mark Word中写入一些标记数据,就是通知他人:这个对象是我的啦!如果其余线程想拜访这个实例的话,就须要进入期待队列,直到以后的线程开释对象的锁,也就是把Mark Word中的数据擦除。

当一个线程领有了锁之后,它便能够屡次进入。当然,在这个线程开释锁的时候,那么也须要执行雷同次数的开释动作。比方,一个线程先后3次取得了锁,那么它也须要开释3次,其余线程才能够持续拜访。

上面的表格展现的是64位计算机中的对象头信息:

|------------------------------------------------------------------------------------------------------------|--------------------||                                            Object Header (128 bits)                                        |        State       ||------------------------------------------------------------------------------|-----------------------------|--------------------||                                  Mark Word (64 bits)                         |    Klass Word (64 bits)     |                    ||------------------------------------------------------------------------------|-----------------------------|--------------------|| unused:25 | identity_hashcode:31 | unused:1 | age:4 | biased_lock:1 | lock:2 |    OOP to metadata object   |       Normal       ||------------------------------------------------------------------------------|-----------------------------|--------------------|| thread:54 |       epoch:2        | unused:1 | age:4 | biased_lock:1 | lock:2 |    OOP to metadata object   |       Biased       ||------------------------------------------------------------------------------|-----------------------------|--------------------||                       ptr_to_lock_record:62                         | lock:2 |    OOP to metadata object   | Lightweight Locked ||------------------------------------------------------------------------------|-----------------------------|--------------------||                     ptr_to_heavyweight_monitor:62                   | lock:2 |    OOP to metadata object   | Heavyweight Locked ||------------------------------------------------------------------------------|-----------------------------|--------------------||                                                                     | lock:2 |    OOP to metadata object   |    Marked for GC   ||------------------------------------------------------------------------------|-----------------------------|--------------------|

从表格中,你能够看到Object Header中的三局部信息:Mark Word、Klass Word、State.

二、通过JOL体验Mark Word的变动

为了直观感触对象头中Mark Word的变动,咱们能够通过 JOL(Java Object Layout) 工具演示一遍。JOL是一个不错的Java内存布局查看工具,心愿你能记住它。

首先,在工程中引入依赖:

<dependency>    <groupId>org.openjdk.jol</groupId>    <artifactId>jol-core</artifactId>    <version>0.10</version></dependency>

在上面的代码中,master是咱们创立的对象实例,办法decreaseBlood()中会执行加锁动作。所以,在调用decreaseBlood()加锁后,对象头信息应该会发生变化

 public static void main(String[] args) {        Master master = new Master();        System.out.println("====加锁前====");        System.out.println(ClassLayout.parseInstance(master).toPrintable());        System.out.println("====加锁后====");        synchronized (master) {            System.out.println(ClassLayout.parseInstance(master).toPrintable());        }    }

后果输入如下:

====加锁前====cn.tao.king.juc.execises1.Master object internals: OFFSET  SIZE   TYPE DESCRIPTION                               VALUE      0     4        (object header)                           01 00 00 00 (00000001 00000000 00000000 00000000) (1)      4     4        (object header)                           00 00 00 00 (00000000 00000000 00000000 00000000) (0)      8     4        (object header)                           43 c1 00 f8 (01000011 11000001 00000000 11111000) (-134168253)     12     4    int Master.blood                              100Instance size: 16 bytesSpace losses: 0 bytes internal + 0 bytes external = 0 bytes total====加锁后====cn.tao.king.juc.execises1.Master object internals: OFFSET  SIZE   TYPE DESCRIPTION                               VALUE      0     4        (object header)                           48 f9 d6 00 (01001000 11111001 11010110 00000000) (14088520)      4     4        (object header)                           00 70 00 00 (00000000 01110000 00000000 00000000) (28672)      8     4        (object header)                           43 c1 00 f8 (01000011 11000001 00000000 11111000) (-134168253)     12     4    int Master.blood                              95Instance size: 16 bytesSpace losses: 0 bytes internal + 0 bytes external = 0 bytes totalProcess finished with exit code 0

从后果中能够看到,代码在执行synchronized办法后,所打印出的object header信息由01 00 00 0000 00 00 00变成了48 f9 d6 0000 70 00 00等等,不出意外的话,置信你应该看不明确这些内容的含意。

所以,为了不便浏览,咱们在青铜系列文章《借花献佛-JOL格式化工具》中提供了一个工具类,让输入更具可读性。借助工具类,咱们把代码调整为:

 public static void main(String[] args) {        Master master = new Master();        System.out.println("====加锁前====");        printObjectHeader(master);        System.out.println("====加锁后====");        synchronized (master) {            printObjectHeader(master);        }    }

输入的后果如下:

====加锁前====# WARNING: Unable to attach Serviceability Agent. You can try again with escalated privileges. Two options: a) use -Djol.tryWithSudo=true to try with sudo; b) echo 0 | sudo tee /proc/sys/kernel/yama/ptrace_scopeClass Pointer: 11111000 00000000 11000001 01000011 Mark Word:    hashcode (31bit): 0000000 00000000 00000000 00000000     age (4bit): 0000    biasedLockFlag (1bit): 0    LockFlag (2bit): 01====加锁后====Class Pointer: 11111000 00000000 11000001 01000011 Mark Word:    javaThread*(62bit,include zero padding): 00000000 00000000 01110000 00000000 00000100 11100100 11101001 100100    LockFlag (2bit): 00

你看,这样一来,输入的后果的后果就高深莫测。从加锁后的后果中能够看到,Mark Word曾经发生变化,以后线程曾经取得对象的锁。

至此,你应该明确,原来synchronized的背地的原理是这么回事。当然,本文所讲述只是其中的局部。出于篇幅思考和难度管制,本文暂且不会对Java对象头中锁的含意和锁的降级等问题开展形容,这部分内容会在前面的文章中具体介绍。

以上就是文本的全部内容,祝贺你又上了一颗星✨

夫子的试炼

  • 下载JOL工具,在代码中体验工具的应用和对象信息的变动。

对于作者

关注公众号【庸人技术笑谈】,获取及时文章更新。记录平凡人的技术故事,分享有品质(尽量)的技术文章,偶然也聊聊生存和现实。不贩卖焦虑,不抛售课程。

如果本文对你有帮忙,欢送点赞关注