当前位置: 首页 > news >正文

synchronized和ReentrantLock的区别

synchronized和ReentrantLock的区别

synchronizedReentrantLock 都是 Java 中用于实现线程同步的机制,但它们在功能、使用方式、性能等方面存在显著区别。以下是两者的详细对比:


1. 基本概念

  • synchronized
    是 Java 内置的关键字,用于实现线程同步,可以修饰方法或代码块。它通过 JVM 实现,属于隐式锁。
  • ReentrantLock
    是 Java 并发包(java.util.concurrent.locks)中的一个类,属于显式锁,提供了比 synchronized 更灵活的锁机制。

2. 主要区别

(1)使用方式
  • synchronized

    • 代码块形式:

      synchronized (lockObject) {// 同步代码块
      }
      
    • 方法形式:

      public synchronized void method() {// 同步方法
      }
      
  • ReentrantLock

    • 需要显式创建锁对象,并在需要时手动加锁和解锁:

      ReentrantLock lock = new ReentrantLock();lock.lock(); // 加锁
      try {// 同步代码
      } finally {lock.unlock(); // 解锁
      }
      
(2)锁的获取与释放
  • synchronized
    • 锁的获取和释放由 JVM 自动管理,无需手动干预。
    • 如果发生异常,JVM 会自动释放锁。
  • ReentrantLock
    • 必须手动调用 lock()unlock() 方法。
    • 必须在 finally 块中释放锁,否则可能导致死锁。
(3)公平性
  • synchronized

    • 默认是非公平锁,无法直接控制锁的公平性。
  • ReentrantLock

    • 可以选择公平锁或非公平锁:

      ReentrantLock fairLock = new ReentrantLock(true); // 公平锁
      ReentrantLock unfairLock = new ReentrantLock(false); // 非公平锁(默认)
      
(4)可中断性
  • synchronized

    • 线程在等待锁时无法被中断,只能一直等待。
  • ReentrantLock

    • 提供了

      lockInterruptibly()
      

      方法,允许线程在等待锁时响应中断:

      javalock.lockInterruptibly(); // 可以被中断
      
(5)尝试获取锁
  • synchronized

    • 没有直接的方法尝试获取锁,只能阻塞等待。
  • ReentrantLock

    • 提供了

      tryLock()
      

      方法,可以尝试获取锁,如果获取失败则立即返回:

      if (lock.tryLock()) {try {// 同步代码} finally {lock.unlock();}
      } else {// 获取锁失败的处理
      }
      
(6)锁绑定多个条件
  • synchronized

    • 只能使用 Objectwait()notify()notifyAll() 方法,且所有线程共享同一个条件队列。
  • ReentrantLock

    • 可以绑定多个

      Condition
      

      对象,实现更精细的线程等待和唤醒:

      Condition condition = lock.newCondition();
      lock.lock();
      try {condition.await(); // 线程等待condition.signal(); // 唤醒线程
      } finally {lock.unlock();
      }
      
(7)性能
  • synchronized
    • 在 Java 6 之前性能较差,但在 Java 6 及之后版本中,JVM 对 synchronized 进行了大量优化(如锁粗化、锁消除、偏向锁、轻量级锁等),性能接近 ReentrantLock
  • ReentrantLock
    • 在高竞争场景下,性能可能优于 synchronized,但需要手动管理锁的获取和释放,增加了代码复杂度。
(8)可读性
  • synchronized
    • 代码更简洁,易于理解和维护。
  • ReentrantLock
    • 代码更复杂,需要手动管理锁的获取和释放,容易出错(如忘记释放锁)。

3. 使用场景

  • synchronized
    • 适用于简单的同步需求,代码简洁,易于维护。
    • 适合不需要复杂锁机制(如公平性、可中断性、多条件)的场景。
  • ReentrantLock
    • 适用于需要更灵活的锁机制(如公平性、可中断性、多条件)的场景。
    • 适合高竞争场景,或需要更精细控制线程同步的场景。

4. 总结对比表

特性synchronizedReentrantLock
使用方式关键字,隐式锁类,显式锁
锁的获取与释放自动管理手动管理(lock()/unlock()
公平性默认非公平可选择公平或非公平
可中断性不支持支持(lockInterruptibly()
尝试获取锁不支持支持(tryLock()
锁绑定多个条件不支持支持(Condition
性能Java 6 后优化,接近 ReentrantLock高竞争场景下可能更优
可读性代码简洁代码复杂

5. 推荐使用

  • 如果不需要复杂的锁机制,优先使用 synchronized,代码更简洁且易于维护。
  • 如果需要公平性、可中断性、多条件等高级功能,或在高竞争场景下需要更精细的控制,使用 ReentrantLock

6. 示例代码

synchronized 示例
public class SynchronizedExample {private int count = 0;public synchronized void increment() {count++;}public static void main(String[] args) {SynchronizedExample example = new SynchronizedExample();Runnable task = () -> {for (int i = 0; i < 1000; i++) {example.increment();}};Thread t1 = new Thread(task);Thread t2 = new Thread(task);t1.start();t2.start();try {t1.join();t2.join();} catch (InterruptedException e) {e.printStackTrace();}System.out.println("Count: " + example.count); // 输出 2000}
}
ReentrantLock 示例
import java.util.concurrent.locks.ReentrantLock;public class ReentrantLockExample {private int count = 0;private final ReentrantLock lock = new ReentrantLock();public void increment() {lock.lock();try {count++;} finally {lock.unlock();}}public static void main(String[] args) {ReentrantLockExample example = new ReentrantLockExample();Runnable task = () -> {for (int i = 0; i < 1000; i++) {example.increment();}};Thread t1 = new Thread(task);Thread t2 = new Thread(task);t1.start();t2.start();try {t1.join();t2.join();} catch (InterruptedException e) {e.printStackTrace();}System.out.println("Count: " + example.count); // 输出 2000}
}

通过以上对比,可以根据具体需求选择合适的同步机制。

相关文章:

  • 【论文阅读】Qwen2.5-VL Technical Report
  • Vue 3 九宫格抽奖系统,采用优雅的 UI 设计和流畅的动画效果
  • 打卡Day53
  • 在tensorrt engine中提高推理性能小记
  • 网络安全防护:Session攻击
  • 【python深度学习】Day53对抗生成网络
  • Vue3 + TypeScript + Element Plus 设置表格行背景颜色
  • 第十七章 Linux之大数据定制篇——Shell编程
  • 【C语言】C语言发展历史、特点及其应用
  • SpringBoot源码解析(十二):@ConfigurationProperties配置绑定的底层转换
  • 树莓派智能小车红外避障实验指导书
  • DeepSeek介绍
  • 【Canvas与曲线】使用贝塞尔二次曲线实现平滑过渡的内旋拼合三角形
  • 人工智能学习13-Numpy-规律数组生成
  • Python训练营-Day31-文件的拆分和使用
  • JAVA实战开源项目:在线课程管理系统 (Vue+SpringBoot) 附源码
  • 【大模型分布式训练】多卡解决单卡训练内存不足的问题
  • 【算法】基于中位数和MAD鲁棒平均值计算算法
  • 全面指南:HTTPX - 下一代Python HTTP客户端
  • 【算法深练】二分答案:从「猜答案」到「精准求解」的解题思路
  • 商城网站制作/广州网站优化运营
  • 海拉尔做网站/seo人员的职责
  • 博客自助建站/抖音推广平台联系方式
  • 滨海专业做网站/企业短视频推广
  • 网站建设中 什么意思/seo专员是干什么的
  • 商务网站建设目的/免费的十大免费货源网站