多线程-synchronized详解

1、基本作用:Java提供了一种内置的锁机制来支持原子性,同步代码块(Synchronized Block )

同步代码块包括两个部分:

  • 1、锁的对象引用。
  • 2、锁作用的代码块。

例子:

synchronized(Object){
    //被保护的共享状态
}

 另外,关键字synchronized修饰的方法,整个方法体就是同步代码块,锁的对象是调用当前方法的对象;需要注意的是,静态方法以对象的字节码文件为(.class文件)作为锁对象。

2、重入

        当某个线程尝试获取其它线程已经持有的锁时,尝试获取锁的线程就会阻塞。但是,内置锁synchronized是可重入的,如果某个线程尝试获取由自己已经持有的锁时,请求成功。

        锁重入表示获取锁的粒度是线程。

重入原理:

(1)编写一个同步方法

public class ClassA{
    Object object = new Object();
    public void method(){
        synchronized (object){

        }
    }
}

(2)对ClassA进行编译和反编译:

编译:

        javac ClassA.java

反编译:

        Javap -verbose ClassA.class

反编译后得到如下:

 每一个对象同一时刻只能关联一个monitor,并且每一个monitor只能在同一时间被一个线程获取,当线程1获取到对象锁之后,线程2只能进入阻塞队列EntryList:

monitor锁计数器会在执行monitorenter/monitorexit指令时+1/-1

  • 当锁的计数器为0时,表示该锁没有被获取,当前线程可以获取,执行monitorenter成功。
  • 如果当前线程获取了锁之后,再次获取,即锁重入,那么在锁的计数器上+1即可,一直重入则一直累加;释放锁之后,锁计数器就-1
  • 如果锁被其他线程拿走了,那就等待释放。

 缺点:

  1. 效率低:释放锁的情况少,不能中断和设置超时。
  2. 不够灵活:加锁和释放锁的时机很单一。
  3. 无法获取锁的状态。

 synchronized锁的升级过程:

  •  无锁:乐观锁就是一种无锁状态。
  • 偏向锁:一个线程多次获取同一把锁,并且没有锁的竞争,通过CAS机制来修改偏向锁的一个标记,如果当前锁时可偏向状态,则判断对象头中的线程ID是否与当前线程ID相同,如果相同,则直接进入。
  • 轻量级锁:其实就是自旋锁,避免将线程直接挂起,从而导致线程状态切换而带来的性能开销。
  • 重量级锁:就是互斥锁,一个线程拿到锁,其他的线程都需要阻塞。当轻量级锁拿不到锁的时候,就会升级为重量级锁。

synchronized锁的升级的思想,是性能和线程安全性之间的平衡,在保证安全性的前提下,最大限度的提升性能。

  • 11
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

何怀逸

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值