1.锁升级
synchronized的加锁过程大约分为以下几部分:
无锁->偏向锁->轻量级锁->重量级锁
这里着重说明一下偏向锁:
偏向锁不是真的加锁,而是一种标志位,它用一个标志记录当前是哪个线程获取的锁,如果这时真的有其他线程也想来获取这把锁,查看标志位发现这把锁已经被占用了,同时锁升级为轻量级锁。这样做的目的就是尽可能的减少加锁带来的资源开销。
2.锁消除
锁消除,即编译器和jvm检测到你加锁的某块代码不涉及线程安全问题,没必要加锁,就自动帮你消除了加锁的步骤。
例如:单线程情况下,使用多线程安全的类,比如StringBuffer的append等等
3.锁粗化
锁粗化是指,你一个线程重复多次的获取释放同一把锁,编译器和jvm就认为你可以一个获取到之后,直到全部事情做完再释放,而不再是做一部分就释放,然后再获取。伪代码如下:
public void fun(){
lock();
//do something1
unlock();
lock();
//do something2
unlock();
lock();
//do something3
unlock();
lock();
//do something4
unlock();
}
--------粗化后:-------
public void fun(){
lock();
//do something1
//do something2
//do something3
//do something4
unlock();
}
这样做的目的就是为了减少多次获取锁和释放锁的开销,使得锁粒度由细变粗。使用细粒度锁, 本来是期望释放锁的时候其他线程能使用锁,但实际检测并没有其他线程来抢占这个锁,那锁就没必要那么细
(锁的粒度变化指的是,锁里面的代码逻辑变化)
结束语:
今天又是一个小细节😗😗