synchronized原理
Posted vycz
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了synchronized原理相关的知识,希望对你有一定的参考价值。
一、概念
是利用锁的机制来实现同步的。
锁机制有如下两种特性:
互斥性:即在同一时间只允许一个线程持有某个对象锁,通过这种特性来实现多线程中的协调机制,这样在同一时间只有一个线程对需同步的代码块(复合操作)进行访问。互斥性我们也往往称为操作的原子性。
可见性:必须确保在锁被释放之前,对共享变量所做的修改,对于随后获得该锁的另一个线程是可见的(即在获得锁时应获得最新共享变量的值),否则另一个线程可能是在本地缓存的某个副本上继续操作从而引起不一致。
二、synchronized的用法
根据修饰对象分类
1、同步方法
(1) 同步非静态方法
Public synchronized void methodName(){
……
}
(2) 同步静态方法
Public synchronized static void methodName(){
……
}
2、同步代码块
synchronized(this|object) {}
synchronized(类.class) {}
Private final Object MUTEX =new Object();
Public void methodName(){
Synchronized(MUTEX ){
……
}
}
根据获取的锁分类
1、获取对象锁
synchronized(this|object) {}
修饰非静态方法
在 Java 中,每个对象都会有一个 monitor 对象,这个对象其实就是 Java 对象的锁,通常会被称为“内置锁”或“对象锁”。类的对象可以有多个,所以每个对象有其独立的对象锁,互不干扰。
2、获取类锁
synchronized(类.class) {}
修饰静态方法
在 Java 中,针对每个类也有一个锁,可以称为“类锁”,类锁实际上是通过对象锁实现的,即类的 Class 对象锁。每个类只有一个 Class 对象,所以每个类只有一个类锁。
在 Java 中,每个对象都会有一个 monitor 对象,监视器。
1) 某一线程占有这个对象的时候,先monitor 的计数器是不是0,如果是0还没有线程占有,这个时候线程占有这个对象,并且对这个对象的monitor+1;如果不为0,表示这个线程已经被其他线程占有,这个线程等待。当线程释放占有权的时候,monitor-1;
2) 同一线程可以对同一对象进行多次加锁,+1,+1,重入性
三、synchronized原理分析
1、线程堆栈分析(互斥)
Jconsole
Jstack pid
2、JVM指令分析
Javap -V 反编译
Monitorenter
Monitorexit
Getstatic
astore_n
aload_n
Monitor
(1)0,lock
(2) 重入
(3) monitor一个线程占有,其他线程请求时会进入BOLCK,直到monitor为0
Monitorexit
计数器减一,为0时为解锁
Javap -v
以上是代码块的加锁monitorenter和monitorExit配合使用
对方法的加锁
ACC_SYNCHRONIZED
1.6JDK以前重量锁
ACC_SYNCHRONIZED
3、使用synchronized注意的问题
(1)与moniter关联的对象不能为空
(2)synchronized作用域太大
(3)不同的monitor企图锁相同的方法
(4)多个锁的交叉导致死锁
Java虚拟机对synchronized的优化
偏向锁
轻量级锁
重量级锁(等待时间长)
对象头与monitor
一个对象实例包含:对象头、实例变量、填充数据
对象头:加锁的基础
无锁状态:没有加锁
偏向锁:在对象第一次被某一线程占有的时候,是否偏向锁置1,锁表01,写入线程号,当其他的线 程访问的时候,竞争,失败 轻量级锁
很 多次悲第一次占有它的线程获取次数多,成功
CAS算法 campany and set(CAS)
无锁状态时间非常接近
竞争不激烈的时候适用
轻量级锁:线程有交替适用,互斥性不是很强,CAS失败,00
重量级锁:强互斥,10,等待时间长
自旋锁:竞争失败的时候,不是马上转化级别,而是执行几次空循环5 10
锁消除:JIT在编译的时候吧不必要的锁去掉
以上是关于synchronized原理的主要内容,如果未能解决你的问题,请参考以下文章