关于Synchronized和ReentranLock的区别及说明
作者:知知之之
文章介绍了Java中的`synchronized`关键字和`ReentrantLock`类,两者都可以用于解决多线程同步问题,但`ReentrantLock`提供了更多的功能和灵活性
Synchronized和ReentranLock的区别
在Java中,synchronized关键字和ReentrantLock类都可以用于解决多线程之间的同步问题,但它们在使用方式和提供的功能上存在一些差异。
Synchronized
synchronized是Java中的一个关键字,作为一种隐式的锁机制,它可以修饰一个方法或者代码块。
使用synchronized时,JVM负责获取和释放锁,使得其使用起来比较简单直接。
- 方法级的同步:当一个方法被
synchronized修饰,那么该方法称为同步方法,同一时间内只有一个线程能执行该方法。 - 代码块级的同步:通过
synchronized修饰代码块,并指定一个锁对象,同一时间内只有持有该锁对象的线程能执行该代码块。
synchronized依赖于Java对象的内置锁,每个对象都可以作为一个锁。
特点:
- 简单易用,自动锁管理(获取锁和释放锁由JVM保证)。
- 不可中断,一旦获得锁,如果不主动释放锁或没有发生异常,则持有锁直到同步块执行完成。
- 不支持公平锁(无法控制锁的获取顺序)。
- 可重入:线程可以重复进入任何一个它已经拥有的锁同步的代码块。
ReentrantLock
ReentrantLock是java.util.concurrent.locks包中的一个类。
与synchronized相比,ReentrantLock提供了更加丰富的锁操作功能,它需要被显式地创建、锁定和释放。
特点:
- 需要程序员手动获取和释放锁(通过
lock()和unlock()方法)。 - 支持公平锁和非公平锁(通过构造函数指定)。
- 提供了一种能够中断等待锁的线程的机制(
lockInterruptibly())。 - 支持获取锁是否成功的条件(通过
tryLock()方法)。 - 支持多个条件变量(
Condition),每个Condition都可以绑定在一个锁上,用来控制线程间的协作。
使用场景对比
- 当同步操作简单,对锁的控制要求不高时,
synchronized是一个很好的选择,因为它可以避免很多由于手动操作锁导致的问题。 - 当需要更高级的线程同步控制,如尝试非阻塞获取锁、要求公平锁、中断等待锁的线程,或者是需要多个条件变量时,
ReentrantLock是更好的选择。
总结
以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。
