synchronized 和 Lock 有什么区别?

synchronized是JVM隐式锁,自动释放,功能简单;Lock是JDK显式锁,需手动释放,支持公平锁、可中断、定时获取等高级功能,灵活性更高。

synchronized 和 Lock 都是 Java 中用于实现线程同步的机制,但它们在使用方式、功能和灵活性上存在显著差异。简而言之,synchronized 是 JVM 提供的关键字,隐式地进行加锁和释放锁,而 Lock 是一个接口,提供了更灵活的锁操作。

synchronized 和 Lock 的区别

1. 实现方式:

  • synchronized: 是 JVM 层面提供的,通过 monitorenter 和 monitorexit 指令来实现。它是一种隐式锁,当 synchronized 代码块或方法执行完毕后,锁会自动释放。
  • Lock: 是 JDK 提供的 API,位于 java.util.concurrent.locks 包中。它是一种显式锁,需要手动加锁(lock())和释放锁(unlock())。

2. 灵活性:

  • synchronized: 功能相对简单,只能实现独占锁(互斥锁),不支持公平锁和非公平锁的选择。
  • Lock: 提供了更多的功能和灵活性,例如:
    • 公平锁和非公平锁: 可以通过 ReentrantLock 的构造方法来指定是否为公平锁。
    • 可中断锁: 可以使用 lockInterruptibly() 方法,允许线程在等待锁的过程中被中断。
    • 定时锁: 可以使用 tryLock(long timeout, TimeUnit unit) 方法,在指定时间内尝试获取锁,如果超时则返回 false
    • 可重入性: ReentrantLock 实现了可重入性,允许同一个线程多次获取同一个锁。

3. 锁的释放:

  • synchronized: 锁的释放是自动的,当 synchronized 代码块或方法执行完毕后,JVM 会自动释放锁。如果发生异常,JVM 也会确保锁被释放。
  • Lock: 锁的释放必须手动进行,需要在 finally 块中调用 unlock() 方法,以确保锁在任何情况下都能被释放。否则,可能会导致死锁。

4. 性能:

  • 在早期的 JDK 版本中,synchronized 的性能通常比 Lock 低。但是,随着 JVM 的优化,synchronized 的性能在某些情况下已经可以与 Lock 相媲美。在 JDK 1.6 之后,synchronized 引入了锁升级的概念(无锁 -> 偏向锁 -> 轻量级锁 -> 重量级锁),使得其性能得到了很大的提升。
  • Lock 提供了更多的性能调优选项,例如公平锁和非公平锁的选择,可以根据具体的应用场景进行优化。

5. 使用场景:

  • synchronized: 适用于简单的同步场景,例如对少量共享资源的访问。由于其简单易用,可以减少代码的编写量。
  • Lock: 适用于复杂的同

    步场景,例如需要实现公平锁、可中断锁、定时锁等高级功能。

应该使用 synchronized 还是 Lock?

选择 synchronized 还是 Lock 取决于具体的应用场景。

  • 如果只需要简单的互斥锁,并且对性能要求不高,那么 synchronized 是一个不错的选择。
  • 如果需要更高级的锁功能,例如公平锁、可中断锁、定时锁等,或者需要更高的性能,那么 Lock 是更好的选择。
  • 需要注意的是,在使用 Lock 时,必须确保在 finally 块中释放锁,以避免死锁。

另外,在考虑性能时,不要过早进行优化。首先编写清晰、可维护的代码,然后根据实际的性能测试结果进行优化。

如何避免死锁?

死锁是指两个或多个线程互相等待对方释放资源,导致所有线程都无法继续执行的情况。以下是一些避免死锁的常用方法:

  1. 避免嵌套锁: 尽量避免在一个线程中同时持有多个锁。如果必须持有多个锁,应该按照固定的顺序获取锁,以避免形成循环等待。

  2. 使用定时锁: 使用 tryLock(long timeout, TimeUnit unit) 方法,在指定时间内尝试获取锁。如果超时,则释放已持有的锁,避免一直等待。

  3. 使用可中断锁: 使用 lockInterruptibly() 方法,允许线程在等待锁的过程中被中断。当线程被中断时,可以释放已持有的锁,避免死锁。

  4. 资源分配图: 使用资源分配图来分析系统中是否存在死锁的可能性。资源分配图是一种图形化的工具,可以帮助识别循环等待的情况。

  5. 死锁检测: 使用死锁检测工具来检测系统中是否存在死锁。死锁检测工具可以定期检查系统的状态,如果发现死锁,则发出警报。

ReentrantLock 的公平锁和非公平锁有什么区别?

ReentrantLock 提供了公平锁和非公平锁两种模式。

  • 公平锁: 按照线程请求锁的顺序来分配锁。如果多个线程同时请求锁,那么先请求锁的线程会先获得锁。公平锁可以避免线程饥饿,但性能相对较低,因为需要维护一个等待队列。

  • 非公平锁: 允许线程插队。当一个线程释放锁时,如果有多个线程正在等待锁,那么 JVM 可以选择任意一个线程来获得锁,而不仅仅是等待队列中的第一个线程。非公平锁的性能通常比公平锁高,但可能会导致某些线程饥饿。

默认情况下,ReentrantLock 使用非公平锁。可以通过 ReentrantLock(boolean fair) 构造方法来指定是否为公平锁。

// 创建一个非公平锁
ReentrantLock lock = new ReentrantLock();

// 创建一个公平锁
ReentrantLock fairLock = new ReentrantLock(true);

选择公平锁还是非公平锁取决于具体的应用场景。如果需要避免线程饥饿,并且对性能要求不高,那么可以使用公平锁。如果对性能要求较高,并且可以容忍一定的线程饥饿,那么可以使用非公平锁。