synchronized关键字是java最重要的关键字 面试多线程必问点之一,那么今天学习了一些内容就简单记录一下
个人学习记录 不成体系 可能有错
java1.6 synchronized优化
java1.6对synchronized做了优化 仅我所知的一点是在 轻量级转重量级锁时做的优化
- 优化前 有线程自旋十次或者竞争线程超过cpu核数的二分之一时转成重量级锁
- 优化后采用自适应方式进入重量级锁
- 其他优化暂时未学到后续补充
锁升级过程
此外synchronized 关键字的代表的锁升级过程如下
无锁->偏向锁->轻量级锁->重量级锁
无锁状态:对象头后三位为: 001
偏向锁: 对象头后三位为 101
轻量级锁:对象头后两位为 10
重量级锁:对象头后两位为 11
锁升级条件
-
无锁
创建对象 无锁状态 -
无锁->偏向锁
当有且仅有一个线程调用该对象完成某种操作 该线程将线程指针记录在对象中 -
偏向锁->轻量级锁
当有两个或者两个以上线程 进行 资源竞争时 此外他们竞争的方式是 CAS(Compare and Swap,即比较再交换。) -
轻量级锁->重量级锁
当竞争更加激烈即发生 有线程自旋十次或者竞争线程超过cpu核数的二分之一时转成重量级锁
轻量级锁和重量级锁的优劣
- 轻量级锁会占用cpu资源 执行速度较快 使用条件 竞争少 线程执行快即是等待时间短
- 重量级锁不会占用cpu资源 使用条件 竞争多 线程执行时间长
volatile关键字作用
- 线程可见性
public class test {
private static volatile boolean flog = true;
public static void main(String[] args) throws InterruptedException {
new Thread(()-> {
while (flog){
//do port
}
}).run();
Thread.sleep(1000);
flog = false;
}
}
此案例 将变量flog 的关键字增删测试 即可理解该作用
- 防止指令重排序
cpu执行多线程时会乱序执行 这样会导致汇编执行的顺序调换 使用volatile关键字可以防止指令的重排序