当前位置: 首页 > 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}
}

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

http://www.xdnf.cn/news/1034263.html

相关文章:

  • gpt3大模型蒸馏后效果会变差么
  • HTTP 请求报文 方法
  • 湖北理元理律师事务所债务优化实务:平衡还款与生活的法律路径
  • 2022mpsPTE岗位笔试题
  • 自动化立体仓库堆垛机控制系统STEP7 FC1功能块 读取位置值SSI接口
  • KJY0047-J1阶段测试
  • 模拟 AJAX 提交 form 表单及请求头设置详解
  • 人工智能学习24-BP激活函数
  • 全球化2.0|云轴科技ZStack联合Teleplex举办技术沙龙 · 吉隆坡站
  • Oracle 单实例双IP配置
  • Qt:Qt桌面程序正常退出注意事项
  • 人工智能学习16-Numpy
  • YOLOv2 训练过程详解:从数据到模型落地的全流程解析
  • 设计模式-创建型模式(详解)
  • 11_13小结
  • 每天一个前端小知识 Day 1
  • 迁移数据库服务器和应用服务器步骤
  • Vue3中v-bind=“$attrs“应用实例
  • 最小费用最大流算法
  • 架构下的最终瓶颈:数据库如何破局?
  • ARDM:一款国产跨平台的Redis管理工具
  • React项目常用目录结构
  • 细节致胜:如何重塑反向海淘用户体验
  • MongoDB 事务有哪些限制和注意事项?
  • 系统学习·PHP语言
  • sqli-labs靶场46-53关(综合)
  • c 语言如何将 uint8_t *tg_pFrames的数据给 uint8_t **ppJpg
  • YOLO11中的C3K2模块
  • AORSA关键文件及参数解释
  • Go语言---闭包