1、基本作用:Java提供了一种内置的锁机制来支持原子性,同步代码块(Synchronized Block )
同步代码块包括两个部分:
- 1、锁的对象引用。
- 2、锁作用的代码块。
例子:
synchronized(Object){
//被保护的共享状态
}
另外,关键字synchronized修饰的方法,整个方法体就是同步代码块,锁的对象是调用当前方法的对象;需要注意的是,静态方法以对象的字节码文件为(.class文件)作为锁对象。
2、重入
当某个线程尝试获取其它线程已经持有的锁时,尝试获取锁的线程就会阻塞。但是,内置锁synchronized是可重入的,如果某个线程尝试获取由自己已经持有的锁时,请求成功。
锁重入表示获取锁的粒度是线程。
重入原理:
(1)编写一个同步方法
public class ClassA{
Object object = new Object();
public void method(){
synchronized (object){
}
}
}
(2)对ClassA进行编译和反编译:
编译:
javac ClassA.java
反编译:
Javap -verbose ClassA.class
反编译后得到如下:
每一个对象同一时刻只能关联一个monitor,并且每一个monitor只能在同一时间被一个线程获取,当线程1获取到对象锁之后,线程2只能进入阻塞队列EntryList:
monitor锁计数器会在执行monitorenter/monitorexit指令时+1/-1
- 当锁的计数器为0时,表示该锁没有被获取,当前线程可以获取,执行monitorenter成功。
- 如果当前线程获取了锁之后,再次获取,即锁重入,那么在锁的计数器上+1即可,一直重入则一直累加;释放锁之后,锁计数器就-1
- 如果锁被其他线程拿走了,那就等待释放。
缺点:
- 效率低:释放锁的情况少,不能中断和设置超时。
- 不够灵活:加锁和释放锁的时机很单一。
- 无法获取锁的状态。
synchronized锁的升级过程:
- 无锁:乐观锁就是一种无锁状态。
- 偏向锁:一个线程多次获取同一把锁,并且没有锁的竞争,通过CAS机制来修改偏向锁的一个标记,如果当前锁时可偏向状态,则判断对象头中的线程ID是否与当前线程ID相同,如果相同,则直接进入。
- 轻量级锁:其实就是自旋锁,避免将线程直接挂起,从而导致线程状态切换而带来的性能开销。
- 重量级锁:就是互斥锁,一个线程拿到锁,其他的线程都需要阻塞。当轻量级锁拿不到锁的时候,就会升级为重量级锁。
synchronized锁的升级的思想,是性能和线程安全性之间的平衡,在保证安全性的前提下,最大限度的提升性能。