synchronized

简介

synchronized可以锁在方法上也可以锁在代码块上
非静态的方法加锁时,锁的是对象,称为对象锁。
静态的方法加锁时,称为类锁

原理

1)synchronized同步代码块的实现是通过monitorenter和monitorexit指令,monitorenter标识开始位置,monitorexit标识结束位置。monitorenter编译后插入同步代码块开始位置,monitorexit插入方法结束处和异常处。当执行monitorenter指令时,线程试图获取锁也就是获取monitor(monitor对象存在于每个]ava对象的对象头中,synchronized锁便是通过这种方式获取锁的,也是为什么Java中任意对象可以作为锁的原因)的持有权。
其内部包含一个计数器,当计数器为0则可以成功获取,获取后将锁计数器设为1。相应的在执行monitorexit指令后,将锁计数器设为0,表明锁被释放。如果获取对象锁失败,那当前线程就要阻塞等待,直到锁被另外一个线程释放为止。

2)synchronized修饰的方法并没有monitorenter指令和monitorexit指令,取得代之的确实是ACC_ SYNCHRONIZED标识,该标识指明了该方法是一个同步方法,JVM通过该ACC. SYNCHRONIZED访问标志来辨别一个方法是否声明为同步方法,从而执行相应的同步调用。

锁升级

synchronized的三种状态,偏向锁,轻量锁,重量锁,不可逆
偏向锁,只有一个线程访问。刚开始的时候线程少,慢慢多起来的,在只有一个线程时使用偏向锁,当线程竞争到个位数时升级到轻量级锁。

轻量级锁,轻量级锁是通过死循环不断去判断有没有资源,没有就继续自旋,并发很多时,自旋消耗很多cpu,导致任务很慢,只适合低并发。自旋次数的默认值是10次,用户可以修改--XX:PreBlockSpin来更改

重量级锁,线程都在就绪队列,会对加锁资源进行竞争,未竞争成功的进入阻塞队列(不会产生额外开销),等线程释放竞争资源后,会通知阻塞队列,重新竞争。cpu会全力执行任务。

锁升级原理

在锁对象的对象头里面有一个threadid的字段,

第一次访问的时候,也就是只有一个线程的时候,threadid为空,jvm会让其持有偏向锁,并将threadid设置为线程id。

再次访问时,会先判断threadid与线程id是否一致,如果一致,可以直接使用此对象,如果不一致,这时候也就是有2个线程,偏向锁就会升级为轻量级锁,通过自旋一定次数来获取锁

执行一定次数之后,还没有正常获取使用对象,这时候cpu空转次数多,资源浪费也很高,轻量级锁升级为重量级锁。

这就是锁的升级过程。

**锁升级目的:**锁升级是为了减低锁带来的性能消耗。

作用

1)原子性,是指一个操作要么全部执行要么全部不执行。

2)可见性,可见性是指多个线程访问一个资源时,该资源的状态、值信息等对于其他线程都是可见的。

3)有序性,synchronized保证了每个时刻都只有一个线程访问同步代码块,也就确定了线程执行同步代码块是分先后顺序的,保证了有序性。

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值