synchronized几个小面试点

synchronized的锁升降级

Java SE1.6为了减少获得锁和释放锁所带来的性能消耗,引入了“偏向锁”和“轻量级锁”,

所以在Java SE1.6里锁一共有四种状态,无锁状态,偏向锁状态,轻量级锁状态和重量级锁状态,它会随着竞争情况逐渐升级。

锁可以升级但不能降级,意味着偏向锁升级成轻量级锁后不能降级成偏向锁。

这种锁升级却不能降级的策略,目的是为了提高获得锁和释放锁的效率。

图解

synchronized几个小面试点_第1张图片

通俗来讲就是:

偏向锁:仅有一个线程进入临界区
轻量级锁:多个线程交替进入临界区(java没有自旋锁的api,轻量级锁即时自旋锁,从1.6开始的)
重量级锁:多个线程同时进入临界区

synchronized的锁作用过程

  1. 获取互斥锁,清空工作内存中的共享变量的值
  2. 在主内存中拷贝最新变量的副本到工作内存
  3. 执行代码
  4. 将更改后的共享变量的值刷新到主内存中
  5. 释放互斥锁

synchronized 能够实现原子性和可见性,本质上依赖的是底层操作系统的互斥锁机制。

双重检查锁的写法

private static volatile SingleMan sInstance; //这里为什么要用volatile呢?因为有些对象在还没初始化完成的时候,对外就已经暴露不为空,但是此时还不能用,如果此时有线程使用了这个对象,就会有问题。加入volatile就可以同步状态

static  SingleMan newInstance(){
  if(sInstance = null){ //可能有两个线程同时到了这个地方,都觉得是空,然后可能会同时去尝试拿monitor,然后另外一个进入等待,当对象初始化后,等待的线程往下走,此时就已经不为空。所以,需要双重检查
    synchroinzed(SingleMan.class){
      if(sInstance = null){
        sInstance= new SingleMan();
       }
    }
  } 
}

你可能感兴趣的:(java)