【Java 并发秘籍】synchronized vs ReentrantLock:揭秘线程同步神器的对决!

简介: 【8月更文挑战第24天】本文详细对比了Java并发编程中`synchronized`关键字与`ReentrantLock`的不同之处。`synchronized`作为内置关键字,提供自动锁管理但不支持中断或公平锁;`ReentrantLock`则通过显式调用方法控制锁,具备更多高级功能如可中断、公平锁及条件变量。文章通过两个计数器类实例展示了两种机制的具体应用,帮助读者理解其差异及适用场景。掌握这两者对于提升多线程程序设计能力至关重要。

在 Java 并发编程中,synchronizedReentrantLock 都是非常重要的同步机制,用于控制多个线程对共享资源的访问。虽然它们都能实现线程同步,但在实际使用中,两者之间存在一些重要的区别。本文将以教程的形式详细介绍 synchronizedReentrantLock 的区别,并通过示例代码展示它们的使用方法。

synchronized 关键字

synchronized 是 Java 中的一个关键字,它可以用于方法或者代码块,用来声明一个同步区域。当一个线程进入 synchronized 修饰的代码块时,它会自动获取锁;当它退出该代码块时,锁也会自动释放。synchronized 锁是 JVM 层面实现的,它具有以下特点:

  • 自动获取与释放锁:不需要显式地获取和释放锁,简化了代码编写。
  • 不可中断:当一个线程持有锁时,其他线程只能等待,即使发生中断也无法打断。
  • 支持重入:同一个线程可以多次获取同一把锁,每次获取锁都必须匹配一次释放。
  • 锁粒度:可以作用于方法级别或代码块级别。
  • 非公平锁:默认情况下,synchronized 锁是非公平的,这意味着它不能保证线程的公平性。

示例代码

以下是一个使用 synchronized 的示例代码,用于实现一个简单的计数器类:

public class Counter {
   
    private int count = 0;

    public synchronized void increment() {
   
        count++;
    }

    public synchronized int getCount() {
   
        return count;
    }
}

public class SynchronizedCounterDemo {
   
    public static void main(String[] args) {
   
        Counter counter = new Counter();
        Thread t1 = new Thread(() -> {
   
            for (int i = 0; i < 10000; i++) {
   
                counter.increment();
            }
        });

        Thread t2 = new Thread(() -> {
   
            for (int i = 0; i < 10000; i++) {
   
                counter.increment();
            }
        });

        t1.start();
        t2.start();

        try {
   
            t1.join();
            t2.join();
        } catch (InterruptedException e) {
   
            e.printStackTrace();
        }

        System.out.println("Final count: " + counter.getCount());
    }
}

ReentrantLock 类

ReentrantLockjava.util.concurrent.locks 包中的一个接口实现,它提供了比 synchronized 更加灵活的锁定机制。ReentrantLock 具有以下特点:

  • 显式获取与释放锁:使用 lock()unlock() 方法显式地获取和释放锁。
  • 可中断:线程在等待锁的过程中可以被中断。
  • 支持公平锁:可以设置为公平锁或非公平锁,默认是非公平锁。
  • 支持条件变量ReentrantLock 提供了条件变量的支持,可以通过 newCondition() 方法创建条件对象。
  • 更细粒度的控制:提供了更多的控制选项,如尝试获取锁、可重入性等。

示例代码

以下是一个使用 ReentrantLock 的示例代码,用于实现一个简单的计数器类:

import java.util.concurrent.locks.ReentrantLock;

public class Counter {
   
    private int count = 0;
    private final ReentrantLock lock = new ReentrantLock();

    public void increment() {
   
        lock.lock();
        try {
   
            count++;
        } finally {
   
            lock.unlock();
        }
    }

    public int getCount() {
   
        lock.lock();
        try {
   
            return count;
        } finally {
   
            lock.unlock();
        }
    }
}

public class ReentrantLockCounterDemo {
   
    public static void main(String[] args) {
   
        Counter counter = new Counter();
        Thread t1 = new Thread(() -> {
   
            for (int i = 0; i < 10000; i++) {
   
                counter.increment();
            }
        });

        Thread t2 = new Thread(() -> {
   
            for (int i = 0; i < 10000; i++) {
   
                counter.increment();
            }
        });

        t1.start();
        t2.start();

        try {
   
            t1.join();
            t2.join();
        } catch (InterruptedException e) {
   
            e.printStackTrace();
        }

        System.out.println("Final count: " + counter.getCount());
    }
}

总结

通过上述教程,我们可以了解到 synchronizedReentrantLock 在 Java 并发编程中的不同之处。虽然两者都能实现线程同步,但在实际使用中,ReentrantLock 提供了更多的灵活性和控制选项,而 synchronized 更加简单易用。选择哪种同步机制取决于具体的应用场景和需求。无论是在日常开发还是面试准备中,熟悉这两种同步机制的区别都是非常重要的。

相关文章
|
5月前
|
JSON 网络协议 安全
【Java】(10)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
293 1
|
5月前
|
JSON 网络协议 安全
【Java基础】(1)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
313 1
|
6月前
|
数据采集 存储 弹性计算
高并发Java爬虫的瓶颈分析与动态线程优化方案
高并发Java爬虫的瓶颈分析与动态线程优化方案
Java 数据库 Spring
281 0
|
Java 开发者
在Java多线程编程的世界里,Lock接口正逐渐成为高手们的首选,取代了传统的synchronized关键字
在Java多线程编程的世界里,Lock接口正逐渐成为高手们的首选,取代了传统的synchronized关键字
210 4
|
存储 安全 Java
Java面试题:请解释Java内存模型,并说明如何在多线程环境下使用synchronized关键字实现同步,阐述ConcurrentHashMap与HashMap的区别,以及它如何在并发环境中提高性能
Java面试题:请解释Java内存模型,并说明如何在多线程环境下使用synchronized关键字实现同步,阐述ConcurrentHashMap与HashMap的区别,以及它如何在并发环境中提高性能
188 0
|
安全 Java 开发者
Java多线程:synchronized关键字和ReentrantLock的区别,为什么我们可能需要使用ReentrantLock而不是synchronized?
Java多线程:synchronized关键字和ReentrantLock的区别,为什么我们可能需要使用ReentrantLock而不是synchronized?
353 0
|
安全 Java 编译器
Java多线程基础-6:线程安全问题及解决措施,synchronized关键字与volatile关键字(一)
线程安全问题是多线程编程中最典型的一类问题之一。如果多线程环境下代码运行的结果是符合我们预期的,即该结果正是在单线程环境中应该出现的结果,则说这个程序是线程安全的。 通俗来说,线程不安全指的就是某一代码在多线程环境下执行会出现bug,而在单线程环境下执行就不会。线程安全问题本质上是由于线程之间的调度顺序的不确定性,正是这样的不确定性,给我们的代码带来了很多“变数”。 本文将对Java多线程编程中,线程安全问题展开详细的讲解。
339 0
|
安全 Java 调度
Java多线程- synchronized关键字总结
Java多线程- synchronized关键字总结
169 0
|
安全 Java 数据安全/隐私保护
Java基础进阶多线程-线程安全和synchronized关键字
Java基础进阶多线程-线程安全和synchronized关键字
Java基础进阶多线程-线程安全和synchronized关键字