1、synchronized 和 volatile的区别
volatile 解决的是内存可见性问题,会使得所有对 volatile 变量的读写都直接写入主存,即保证了变量的可见性。 synchronized 解决的是执行控制的问题,他会阻止其他线程获取当前对象的监控锁,这样一来就让当前对象被 synchronized 关键字保护的代码无法被其他线程访问,也就是无法并发执行。synchronized 会创建一个内存屏障,内存屏障指令保证了所有CPU操作结果都会直接刷到主存中,从而保证操作的内存可见性,同时也使得这个锁的线程的所有操作都 happens-before 于随后获得这个锁的线程的操作。
两者的区别主要有如下:
- volatile 本质是在告诉JVM当前变量在寄存器(工作内存)中的是不确定的,需要从主存中读取;synchronized 则是锁定当前变量,只有当前线程可以访问该变量,其他线程被阻塞住。
- volatile 仅能使用在变量级别;synchronized 则可以使用在变量、方法和类级别的。
- volatile 仅能实现变量的修改可见性,不能保证原子性;而 synchronized 则可以保证变量的修改可见性和原子性。
- volatile 不会造成线程的阻塞;synchronized 可能会造成线程的阻塞。
- volatile 标记的变量不会被编译器优化;synchronized 标记的变量可以被编译器优化。
2、synchronized 和 Lock 的区别
- synchronized 可以给类、方法、代码块加锁;而Lock只能给代码块加锁。
- synchronized 不需要手动获取锁和释放锁,使用简单,发生异常会自动释放锁,不会造成死锁。
- 通过Lock 可以知道有没有成功获取锁,而 synchronized 却无法办到。
3、synchronized 和 ReentrantLock 的区别
- 两者都是可重入锁。
- synchronized 依赖于 JVM 而 ReentrantLock 依赖于 API。
- ReentrantLock 比 synchronized 增加了一些高级功能。
- 等待可中断。通过 Lock.lockInterruptibly() 来实现这个机制。也就是说正在等待的线程可以选择放弃等待,改为处理其他事情。
- ReentrantLock 可以指定是公平锁还是非公平锁。而synchronized只能是非公平锁。
- 用 ReentrantLock 类结合 Condition 实例可以实现“选择性通知”。
使用选择:
- 除非需要使用 ReentrantLock 的高级功能,否则优先使用 synchronized。
- synchronized 是JVM实现的一种锁机制,JVM 原生地支持它,而ReentrantLock 不是所有的 JDK 版本都支持。并且使用 synchronized 不用担心没有释放锁而导致死锁问题,因为 JVM 会确保锁的释放。
|