synchronized理解

54 阅读2分钟

持续创作,加速成长!这是我参与「掘金日新计划 · 6 月更文挑战」的第19天,点击查看活动详情

背景

  1. 理解并记忆实例对象布局
  2. 理解并记忆对象头布局
  3. 对象的状态

过程

  • 实例对象布局 在这里插入图片描述
  1. 对象头(mark word, class pointer)
  2. 实例数据
  3. 对齐填充(保证是8个字节的整数倍,方便机器加载或者读取字节到内存或者寄存器)
  • 对象头布局
  1. mark word : 对象哈希码,对象分代年龄,指向锁记录的指针,指向重量级锁记录的指针,GC标记,偏向线程ID,偏向时间戳

  2. klass pointer: 类型指针。 对象指向它的类元数据的指针,虚拟机通过这个指针来确定这个对象是哪个类的实例。

  • 普通对象实例和数组对象实例布局有差异。数组对象实例布局多了一个数组的长度

  • 对象的状态

    无锁, 偏向锁, 轻量级锁, 重量级锁, GC标记

  • 测试synchronized加锁过程

  1. 测试代码

    public class HelloWorld implements Runnable{
        private static HelloWorld helloWorld = new HelloWorld();
        int a;
        float b;
        boolean c;
        public static void main(String[] args) {
            System.out.println(ClassLayout.parseInstance(helloWorld).toPrintable());
            new Thread(helloWorld).start();
        }
    
        @Override
        public void run() {
            synchronized (helloWorld) {
                System.out.println(ClassLayout.parseInstance(helloWorld).toPrintable());
            }
        }
    }
    
  2. 测试结果 在这里插入图片描述

  • synchronized支持可重入。

  • synchronized,如果出现异常,未捕获,则释放锁。如果捕获,则不释放锁。

  • synchronized可保证原子性。volatile不可保证原子性。

  • wait,释放锁。notify,不释放锁。

小结

  1. synchronized关键字上锁的时候,修改了对象头哪些信息。
  2. 小结,synchronized关键字其实是对对象进行上锁的。比如用在方法上,其实就是对当前对象this这个关键字进行了加锁。如果方法是静态的,其实就是对class对象进行了加锁。