锁的分类
java中锁按竞争状态、竞争程度可以分为4种锁,分别为无锁、偏向锁、轻量级锁、重量级锁,这些锁只能按照竞争激烈程度自动升级,不能降级,以提高锁的获取与释放效率。
在java中,对象在内存中存储的布局可以分为三块区域:对象头(Header)、实例数据(Instance Data)和对齐填充(Padding)。在对象头的MarkWord字段中,存储着对象的Hashcode(地址映射),分代年龄,以及锁的标记。
锁的标记一共有四种状态:
无锁 | 0 01 |
---|---|
偏向锁 | 1 01 -线程id、Epoch字段 |
轻量级锁 | 00 |
重量级锁 | 10 |
以及垃圾回收GC标记 | 11 |
偏向锁
偏向锁是所有加锁中最轻量级的,在偏向锁的概念中,锁不但不存在多线程间的竞争,而且从始至终都是同一个线程多次获取同一锁,为了降低线程获取锁的开销所以引入偏向锁。所以在线程访问偏向锁的过程中,只需要第一次请求访问时进行一次CAS操作,此后只要锁持有线程和当前访问线程一致就可以直接拿锁。所以偏向锁要存持有锁线程的id和Epoch表示此偏向锁撤销的次数。
偏向锁的获取与释放过程
1.首先当一个线程访问同步块并CAS成功获取到锁时,会在对象头和栈帧中的锁记录字段存储持有该偏向锁的线程id,以后只要该线程进入或者退出代码块都不在需要CAS进行加