synchronized的锁升降级
Java SE1.6为了减少获得锁和释放锁所带来的性能消耗,引入了“偏向锁”和“轻量级锁”,
所以在Java SE1.6里锁一共有四种状态,无锁状态,偏向锁状态,轻量级锁状态和重量级锁状态,它会随着竞争情况逐渐升级。
锁可以升级但不能降级,意味着偏向锁升级成轻量级锁后不能降级成偏向锁。
这种锁升级却不能降级的策略,目的是为了提高获得锁和释放锁的效率。
图解
通俗来讲就是:
偏向锁:仅有一个线程进入临界区 轻量级锁:多个线程交替进入临界区(java没有自旋锁的api,轻量级锁即时自旋锁,从1.6开始的) 重量级锁:多个线程同时进入临界区
synchronized的锁作用过程
- 获取互斥锁,清空工作内存中的共享变量的值
- 在主内存中拷贝最新变量的副本到工作内存
- 执行代码
- 将更改后的共享变量的值刷新到主内存中
- 释放互斥锁
synchronized 能够实现原子性和可见性,本质上依赖的是底层操作系统的互斥锁机制。
双重检查锁的写法
private static volatile SingleMan sInstance; //这里为什么要用volatile呢?因为有些对象在还没初始化完成的时候,对外就已经暴露不为空,但是此时还不能用,如果此时有线程使用了这个对象,就会有问题。加入volatile就可以同步状态
static SingleMan newInstance(){
if(sInstance = null){ //可能有两个线程同时到了这个地方,都觉得是空,然后可能会同时去尝试拿monitor,然后另外一个进入等待,当对象初始化后,等待的线程往下走,此时就已经不为空。所以,需要双重检查
synchroinzed(SingleMan.class){
if(sInstance = null){
sInstance= new SingleMan();
}
}
}
}
扫描二维码,关注公众号“猿必过”
回复 “面试题” 自行领取吧。
微信群交流讨论,请添加微信号:zyhui98,备注:面试题加群
本文由猿必过 YBG 发布 禁止未经授权转载,违者依法追究相关法律责任 如需授权可联系:zhuyunhui@yuanbiguo.com