Java Synchronized实现原理
阅读数:69 评论数:0
跳转到新版页面分类
python/Java
正文
Java对象头和Monitor是实现synchronized的基础!
一、Java对象头
对象头主要包括两部分数据:Mark Word(标记字段)、Klass Pointer(类型指针)。
1、Klass Pointer
是对象指向它的类元数据的指针,虚拟机通过个指针来确定这个对象是哪个类的实例。
2、Mark Word
用于存储对象自身的运行时数据,如HashCode、GC分代年龄、锁状态标志、线程持有的锁、偏向线程ID、偏向时间戳等等
二、Monitor
我们可以把它理解为同步工具,也可以描述为一种同步机制,它通常被描述为对象监视器。
当多个线程同时请求某个对象监视器时,对象监视器会设置几种状态用来区分请深圳市的线程。
Contention List | 所有请求锁的线程将被首先放置到该竞争队列 |
Entry List | Contention List中那些有资格成为候选人的线程被移到Entry List |
Wait Set | 那些调用wait方法被阻塞的线程被放置到Wait Set |
OnDeck | 任何时刻最多只能有一个线程正在竞争锁,该线程称为OnDeck |
Owner | 获得锁的线程称为Owner |
!Owner | 释放锁的线程 |
新请求锁的线程将首先被加入到ConetentionList中,当某个拥有锁的线程(Owner状态)调用unlock之后,如果发现EntryList为空则从ContentionList中移动线程到EntryList.
1、ContentionList 虚拟队列
ContentionList并不是一个真正的Queue,而只是一个虚拟队列,原因在于ContentionList是由Node及其next指针逻辑构成,并不存在一个Queue的数据结构。ContentionList是一个先进先出(FIFO)的队列,每次新加入Node时都会在队头进行,通过CAS改变第一个节点的的指针为新增节点,同时设置新增节点的next指向后续节点,而取得操作则发生在队尾。显然,该结构其实是个Lock-Free的队列。
因为只有Owner线程才能从队尾取元素,也即线程出列操作无争用,当然也就避免了CAS的ABA问题。
因为只有Owner线程才能从队尾取元素,也即线程出列操作无争用,当然也就避免了CAS的ABA问题。
2、EntryList
EntryList与ContentionList逻辑上同属等待队列,ContentionList会被线程并发访问,为了降低对ContentionList队尾的争用,而建立EntryList。Owner线程在unlock时会从ContentionList中迁移线程到EntryList,并会指定EntryList中的某个线程(一般为Head)为Ready(OnDeck)线程。Owner线程并不是把锁传递给OnDeck线程,只是把竞争锁的权利交给OnDeck,OnDeck线程需要重新竞争锁。这样做虽然牺牲了一定的公平性,但极大的提高了整体吞吐量,在Hotspot中把OnDeck的选择行为称之为“竞争切换”。OnDeck线程获得锁后即变为owner线程,无法获得锁则会依然留在EntryList中,考虑到公平性,在EntryList中的位置不发生变化(依然在队头)。如果Owner线程被wait方法阻塞,则转移到WaitSet队列;如果在某个时刻被notify/notifyAll唤醒,则再次转移到EntryList。
相关推荐
一、概述
Synchronized是基于底层操作系统的Mutex Lock实现的,每次获取锁和释放锁都会带来用户太和内核态的切换,从而增加系统性能开锁,从JDK1.6开始,Java对Synchroni
用法不同
synchronized 可以用来修饰普通方法、静态方法和代码块,而 ReentrantLock 只能用于代码块。
获取锁和释放锁的机制不同
synchronized 是自动加锁和
1、直接常量
为了编译器可以准确的知道要生成什么样的类型,可以给直接常量后面添加后缀字符标志它的类型,若为L表示long,F表示float,D表示double。也可以利用前缀表示进制,0x表示十六进制
一、类的继承
1、说明
(1)extends关键字用于类的继承。
(2)在C++中,方法的动态绑定是使用virtual关键字来实现的,而在Java中,动态绑定是默认的形为,不需要添加额外的关键字。
(
一、创建线程
创建线程有四种方式:继承Thread类、实现Runnable接口、实现Callable接口、通过线程池创建。
1、继承Thread
重写run方法。
class A extends Th
一、Collection接口
Collection接口的iterator和toArray方法都用于获得集合中的“所有元素”。前者返回一个“iterator”对象,后者返回一个包含集合中所有元素的数组。