Java多线程的锁机制
上下文:程序运行需要的环境(外部变量)
上下文切换:将之前的程序需要的外部变量复制保存,然后切换到新的程序运行环境
系统调用:(用户态陷入操作系统,通过操作系统执行内核态指令,执行完回到用户态)用户态——内核态——用户态:两次上下文切换
线程wait()方法:将自身加入等待队列,发生了一次上下文切换
notify()方法:将线程唤醒,也发生了上下文切换
Java线程中的锁:偏向锁、轻量级锁、重量级锁。
注意:偏向锁和轻量级锁都没有发生竞争,重量级锁发生了竞争。
偏向锁:可重入和经常使用某一个线程。
轻量级锁:线程之间的切换,但是未发生竞争(在一个时间段只有一个线程使用)
重量级锁:线程时间的竞争。
临界区:多个线程共享临界区,而且存在对临界区的读写。
无所状态:hashcode 31位 年龄age:4位 是否是偏向锁:block_lock:一位:0表示不是偏向锁
偏向锁:thread:54位 线程id epoch 2位,作为批量重偏向的记录 biased_lock:1 表示为偏向锁
轻量级锁:ptr_to_lock_record:62位 表示,锁记录的指针
重量级锁:ptr_to_heavyweight_monitor :62位 表示monitor指针
轻量级锁的流程:
锁记录有两个指针:一个指向自身,一个指向lock对象,当lock对象位state位01时,表示位轻量级锁,此时执行轻量级锁的加锁:交换lock record和markword
锁重入:
同样创建一个LockRecord 但此时cas失败,因为此时lock对象的markword为当前锁记录的指针,置lockrecord指针为null,表示锁重入,并且添加一个锁重入计数器,记录锁重入的次数
解锁:将LockRecord中的lockrecord(此时是markword)和lock对象(Object)中的markword(lockrecord指针)。如果解锁失败,那么轻量级锁就会膨胀为重量级锁。
重量级锁的加锁、解锁、阻塞、唤醒:只有重量级锁有阻塞
如上图:当发生竞争时,新的线程会将object修改为monitor指针,指向重量级锁的monitor(管程),然后进入等待队列,当当前线程结束同步代码块,就会将owner置为0,此时表示没有线程拥有此锁,然后唤醒其他线程,其他线程就会竞争这个锁。
偏向锁的加锁和解锁:新的线程只需要判断lock对象的markword对象中的线程id是否是自己的id,如果是,那么就直接使用这个锁。不用作cas交换,只有第一次获得此锁时需要用cas交换
调用hashcode可以撤销偏向锁,或者对同一个对象发生锁同步时也会撤销偏向锁
偏向锁重定向:对于一个类的多个实现类锁对象,当超过20次重定向时,后面该锁的实例对象就变成了另外一个线程的偏向锁
当发生40次重定向时,该类的所有lock对象就变成了轻量级锁。