synchronized 超详解

synchronized 详解



前言

synchronized用法

1.修饰普通方法:作用于当前对象实例,进入同步代码前要获得当前对象实例的锁

public synchronized void method() {  
    // ...  
}

2.修饰静态方法:作用于当前类,进入同步代码前要获得当前类对象的锁,synchronized关键字加到static 静态方法和 synchronized(class)代码块上都是是给 Class 类上锁

3.修饰代码块:指定加锁对象,对给定对象加锁,进入同步代码库前要获得给定对象的锁

public void method() {  
    synchronized (this) {  
        // ...  
    }  
}

synchronized 的作用

原子性:确保线程互斥的访问同步代码;

可见性:保证共享变量的修改能够及时可见;

有序性:有效解决重排序问题。

synchronized的底层原理

synchronized 同步代码块的实现是通过 monitorentermonitorexit 指令,其中 monitorenter 指令指向同步代码块的开始位置,monitorexit 指令则指明同步代码块的结束位置。

当执行 monitorenter 指令时,线程试图获取锁也就是获取 monitor的持有权(monitor对象存在于每个Java对象的对象头中, synchronized 锁便是通过这种方式获取锁的,也是为什么Java中任意对象可以作为锁的原因)。

其内部包含一个计数器,当计数器为0则可以成功获取,获取后将锁计数器设为1也就是加1。相应的在执行 monitorexit 指令后,将锁计数器设为0 ,表明锁被释放。

如果获取对象锁失败,那当前线程就要阻塞等待,直到锁被另外一个线程释放为止synchronized 修饰的方法并没有 monitorenter 指令和 monitorexit 指令,取得代之的确实是ACC_SYNCHRONIZED 标识,该标识指明了该方法是一个同步方法,JVM 通过该 ACC_SYNCHRONIZED 访问标志来辨别一个方法是否声明为同步方法,从而执行相应的同步调用。

  • 内部锁(Intrinsic Lock):synchronized 关键字在 JVM 层面上是围绕内部锁或监管锁(Monitor Lock)的实体建立的。Java 利用锁机制实现线程同步的一种方式。
  • 隐式锁:synchronized 属于隐式锁,与显式锁(如 ReentrantLock)不同,它不需要程序员手动获取和释放锁。
  • 可重入锁:当一个线程已经持有一个对象的锁时,它可以再次请求该对象的锁,而不会造成死锁。这是因为 synchronized 的锁对象中有一个计数器会记录线程获得锁的次数。
  • 非公平锁:多个线程尝试获取锁时,会直接去尝试获取,如果获取不到则进入等待队列。

synchronized的优缺点

优点:

  • 简单易用:只需要在方法或代码块前加上 synchronized 关键字即可。
  • 隐式锁:不需要手动获取和释放锁。
  • 可重入锁:避免死锁

缺点

  • 灵活性差:synchronized 只能控制对同一对象的访问,不能控制对不同对象的访问。
  • 效率低:相对于显式锁,synchronized 的性能较差。
  • 不可中断:当一个线程持有锁时,其他线程必须等待该线程释放锁,不可被中断。

总结

synchronized 是 Java 中实现线程同步的一种重要方式,它可以修饰方法或代码块。通过内部锁机制,它可以确保同一时刻只有一个线程可以访问共享资源,从而避免了并发问题。但是,它也存在一些缺点,如灵活性差、效率低和不可中断等。

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值