一、加锁对象
- 修饰方法,默认的锁对象就是当前方法的对象实例。
- 静态方法,其锁对象就是此方法所对应的类Class对象。
- 修饰代码块,指定加锁对象,进入同步代码块需要用获得给定对象锁
Public class Test{
public Object lockObj = new Object();
public static void main(String a[]){
Test instanceA = new Test();
testA.func1();
}
public synchronize void func1(){
//方法上加锁,锁住的是当前Test类的实例对象instanceA
//...
}
public synchronize static void func2(){
//静态方法上加锁,锁住的是当前Test类对象(Test.class对象)
//...
}
public void func2(){
//代码块上加锁,锁住的是括号内的对象实例(lockObj对象)
synchronize(lockObj){
//...
}
//...
}
}
二、加锁过程
Synchronized锁分为无锁、偏向锁、轻量级锁、重量级锁,会根据锁竞争情况逐步进线锁升级
1、偏向锁不是真的加锁, 而只是在锁的对象头中记录一个标记,(记录该锁所属的线程),如果没有其他线程参与竞争锁,那么就不会真正执行加锁操作,从而降低程序开销,一旦真的涉及到其他的线程竞争,再取消偏向锁状态,进入轻量级锁状态
2、轻量级锁:其他线程进入竞争,偏向锁状态被消除,进入轻量级锁状态 (自适应的自旋锁),自旋操作是一直让 CPU 空转, 比较浪费 CPU 资源,因此此处的自旋不会一直持续进行, 而是达到一定的时间/重试次数, 就不再自旋了,锁就会升级为重量级锁
3、重量级锁:线程多次自旋或长时间拿不到锁就不再自旋了,而是进入阻塞队列等待锁被其他线程释放,然后被重新唤醒
所以当我们使用Synchronized加锁时,如果当前没有其他线程竞争,锁始终保持偏向锁状态,如果有其他线程竞争,它机会根据竞争激烈程度升级为轻量级锁或者重量级锁