Java 8 中 ReentrantLock 与 Synchronized 的区别

简介: Java 8 中 ReentrantLock 与 Synchronized 的区别

引言


在Java多线程编程中,确保线程安全是至关重要的。为了实现线程的同步和协作,Java提供了多种同步机制,其中最常见的是使用 synchronized 关键字和 ReentrantLock 类。这两者在实现线程安全的同时,也存在着一些区别。本文将深入讨论这两种同步机制的差异,并结合 Java 8 的特性,解析它们在常见面试题中的应用。


同步机制简介


Synchronized

13.pngsynchronized 是Java中最早引入的同步机制,通过对方法或代码块加锁来确保多线程环境下的数据一致性。它使用起来简单,不需要手动释放锁,JVM会自动管理。

public synchronized void synchronizedMethod() {
    // 同步代码块
    // ...
}

ReentrantLock


ReentrantLock 是Java 5 中引入的同步机制,相较于 synchronized,它提供了更灵活的锁定方式。通过显式地获取锁和释放锁,程序员可以更精细地控制同步操作。此外,ReentrantLock 还支持可重入性,即一个线程可以多次获取同一个锁。

import java.util.concurrent.locks.ReentrantLock;
public class MyLock {
    private final ReentrantLock lock = new ReentrantLock();
    public void lockedMethod() {
        lock.lock();
        try {
            // 同步代码块
            // ...
        } finally {
            lock.unlock();
        }
    }
}

区别分析


可中断性


一个显著的区别是 ReentrantLock 支持中断等待锁的线程,而 synchronized 不支持。在 ReentrantLock 中,线程可以通过 lockInterruptibly 方法等待锁,如果其他线程中断了当前线程,它可以响应中断而不是一直等待。

ReentrantLock lock = new ReentrantLock();
public void interruptibleMethod() throws InterruptedException {
    lock.lockInterruptibly();
    try {
        // 同步代码块
        // ...
    } finally {
        lock.unlock();
    }
}

公平性


ReentrantLock 可以选择是否公平地获取锁,而 synchronized 是非公平的。在公平模式下,等待时间最长的线程会优先获得锁。在非公平模式下,线程有一定几率在等待队列为空时插队成功。

ReentrantLock fairLock = new ReentrantLock(true); // 公平锁
ReentrantLock unfairLock = new ReentrantLock();     // 非公平锁

条件变量


ReentrantLock 提供了 Condition 接口,可以通过 newCondition 方法创建多个条件变量,用于在不同的情况下等待或唤醒线程。而 synchronized 只能通过 Object 的 wait、notify 和 notifyAll 方法来实现简单的线程协作,缺乏 ReentrantLock 中的灵活性。

ReentrantLock lock = new ReentrantLock();
Condition condition = lock.newCondition();
public void awaitSignal() throws InterruptedException {
    lock.lock();
    try {
        condition.await();
    } finally {
        lock.unlock();
    }
}
public void sendSignal() {
    lock.lock();
    try {
        condition.signal();
    } finally {
        lock.unlock();
    }
}

性能比较


在 Java 5 之前,synchronized 的性能较差,但在 Java 6 及以后的版本中,对 synchronized 进行了优化,性能已经有了很大的提升。一般来说,在性能上,synchronized 与 ReentrantLock 的差距并不明显。在选择使用时,更应该考虑到代码的可读性和维护性。


Java 8 的新特性


Java 8 引入了函数式编程的特性,同时也对并发编程做出了一些改进。其中一个显著的改变是引入了新的 StampedLock 类,它是 ReentrantLock 的进一步扩展,提供了乐观锁机制,使得在读多写少的场景下性能更优。

import java.util.concurrent.locks.StampedLock;
public class MyStampedLock {
    private final StampedLock lock = new StampedLock();
    public void read() {
        long stamp = lock.tryOptimisticRead();
        // 乐观读操作
        if (!lock.validate(stamp)) {
            // 如果发生了写操作,则使用悲观读锁
            stamp = lock.readLock();
            try {
                // 悲观读操作
            } finally {
                lock.unlockRead(stamp);
            }
        }
    }
    public void write() {
        long stamp = lock.writeLock();
        try {
            // 写操作
        } finally {
            lock.unlockWrite(stamp);
        }
    }
}

面试题解析


什么时候使用 ReentrantLock 而不是 synchronized?


需要支持可中断等待: 如果你的线程需要响应中断而不是一直等待锁,那么 ReentrantLock 是更好的选择。


需要尝试获取锁: ReentrantLock 提供了 tryLock 方法,可以尝试获取锁而不一直等待,可以用于避免死锁。


需要选择公平性: 如果你需要控制线程获取锁的顺序,可以选择 ReentrantLock 的公平锁或非公平锁。

ReentrantLock 的可重入性是什么意思?


可重入性是指同一个线程在持有锁的情况下,能够再次获取这个锁而不发生死锁。ReentrantLock 和 synchronized 都是可重入的,这使得线程可以递归地调用同步方法或同步代码块。


什么是乐观读锁?StampedLock 有什么优势?


乐观读锁是 StampedLock 引入的特性,它允许多个线程同时读取共享资源,而不会阻塞写锁。如果在乐观读锁期间没有写锁被获取,读操作可以立即进行。当然,如果写锁被获取,就需要转为悲观读锁。


StampedLock 的优势在于在读多写少的场景下性能更优,因为它允许多个线程同时读取,而不阻塞彼此。


总结


在Java 8中,ReentrantLock 和 synchronized 依然是同步机制中的主流选择。选择使用哪一种取决于具体的需求和场景。总体而言,synchronized 在简单场景下易用性更高,而 ReentrantLock 则在一些复杂场景下提供了更多的灵活性和控制权。


对于面试而言,了解这两者的区别、使用场景以及新特性,可以展现出对多线程编程深入理解的能力,为应对面试中的技术问题提供有力支持。在实际应用中,结合业务需求和性能要求,选择适当的同步机制是至关重要的。


目录
打赏
0
0
0
0
44
分享
相关文章
【原理】【Java并发】【synchronized】适合中学者体质的synchronized原理
本文深入解析了Java中`synchronized`关键字的底层原理,从代码块与方法修饰的区别到锁升级机制,内容详尽。通过`monitorenter`和`monitorexit`指令,阐述了`synchronized`实现原子性、有序性和可见性的原理。同时,详细分析了锁升级流程:无锁 → 偏向锁 → 轻量级锁 → 重量级锁,结合对象头`MarkWord`的变化,揭示JVM优化锁性能的策略。此外,还探讨了Monitor的内部结构及线程竞争锁的过程,并介绍了锁消除与锁粗化等优化手段。最后,结合实际案例,帮助读者全面理解`synchronized`在并发编程中的作用与细节。
54 8
【原理】【Java并发】【synchronized】适合中学者体质的synchronized原理
|
23天前
|
【Java并发】【synchronized】适合初学者体质入门的synchronized
欢迎来到我的Java线程同步入门指南!我不是外包员工,梦想是写高端CRUD。2025年我正在沉淀中,博客更新速度加快,欢迎点赞、收藏、关注。 本文介绍Java中的`synchronized`关键字,适合初学者。`synchronized`用于确保多个线程访问共享资源时不会发生冲突,避免竞态条件、保证内存可见性、防止原子性破坏及协调多线程有序访问。
53 8
【Java并发】【synchronized】适合初学者体质入门的synchronized
Java 高级面试技巧:yield() 与 sleep() 方法的使用场景和区别
本文详细解析了 Java 中 `Thread` 类的 `yield()` 和 `sleep()` 方法,解释了它们的作用、区别及为什么是静态方法。`yield()` 让当前线程释放 CPU 时间片,给其他同等优先级线程运行机会,但不保证暂停;`sleep()` 则让线程进入休眠状态,指定时间后继续执行。两者都是静态方法,因为它们影响线程调度机制而非单一线程行为。这些知识点在面试中常被提及,掌握它们有助于更好地应对多线程编程问题。
115 9
Java面试必问!run() 和 start() 方法到底有啥区别?
在多线程编程中,run和 start方法常常让开发者感到困惑。为什么调用 start 才能启动线程,而直接调用 run只是普通方法调用?这篇文章将通过一个简单的例子,详细解析这两者的区别,帮助你在面试中脱颖而出,理解多线程背后的机制和原理。
98 12
Java多线程——synchronized、volatile 保障可见性
Java多线程中,`synchronized` 和 `volatile` 关键字用于保障可见性。`synchronized` 保证原子性、可见性和有序性,通过锁机制确保线程安全;`volatile` 仅保证可见性和有序性,不保证原子性。代码示例展示了如何使用 `synchronized` 和 `volatile` 解决主线程无法感知子线程修改共享变量的问题。总结:`volatile` 确保不同线程对共享变量操作的可见性,使一个线程修改后,其他线程能立即看到最新值。
深入理解Java中synchronized三种使用方式:助您写出线程安全的代码
`synchronized` 是 Java 中的关键字,用于实现线程同步,确保多个线程互斥访问共享资源。它通过内置的监视器锁机制,防止多个线程同时执行被 `synchronized` 修饰的方法或代码块。`synchronized` 可以修饰非静态方法、静态方法和代码块,分别锁定实例对象、类对象或指定的对象。其底层原理基于 JVM 的指令和对象的监视器,JDK 1.6 后引入了偏向锁、轻量级锁等优化措施,提高了性能。
118 3
Java社招面试题:& 和 && 的区别,HR的套路险些让我翻车!
今日分享的主题是如何区分&和&&的区别,提高自身面试的能力。主要分为以下四部分。 1、自我面试经历 2、&amp和&amp&amp的不同之处 3、&对&&的不同用回答逻辑解释 4、彩蛋
Java社招面试题:& 和 && 的区别,HR的套路险些让我翻车!
小米,29岁程序员,分享了一次面试经历,详细解析了Java中&和&&的区别及应用场景,展示了扎实的基础知识和良好的应变能力,最终成功获得Offer。
120 14
|
4月前
|
java中面向过程和面向对象区别?
java中面向过程和面向对象区别?
49 1
java 中操作字符串都有哪些类,它们之间有什么区别
Java中操作字符串的类主要有String、StringBuilder和StringBuffer。String是不可变的,每次操作都会生成新对象;StringBuilder和StringBuffer都是可变的,但StringBuilder是非线程安全的,而StringBuffer是线程安全的,因此性能略低。
148 8
AI助理

你好,我是AI助理

可以解答问题、推荐解决方案等