【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 更加简单易用。选择哪种同步机制取决于具体的应用场景和需求。无论是在日常开发还是面试准备中,熟悉这两种同步机制的区别都是非常重要的。

相关文章
|
2月前
|
安全 算法 Java
Java 多线程:线程安全与同步控制的深度解析
本文介绍了 Java 多线程开发的关键技术,涵盖线程的创建与启动、线程安全问题及其解决方案,包括 synchronized 关键字、原子类和线程间通信机制。通过示例代码讲解了多线程编程中的常见问题与优化方法,帮助开发者提升程序性能与稳定性。
117 0
|
2月前
|
Java API 调度
从阻塞到畅通:Java虚拟线程开启并发新纪元
从阻塞到畅通:Java虚拟线程开启并发新纪元
277 83
|
2月前
|
存储 Java 调度
Java虚拟线程:轻量级并发的革命性突破
Java虚拟线程:轻量级并发的革命性突破
222 83
|
10月前
|
Java 开发者
在Java多线程编程的世界里,Lock接口正逐渐成为高手们的首选,取代了传统的synchronized关键字
在Java多线程编程的世界里,Lock接口正逐渐成为高手们的首选,取代了传统的synchronized关键字
114 4
|
存储 安全 Java
Java面试题:请解释Java内存模型,并说明如何在多线程环境下使用synchronized关键字实现同步,阐述ConcurrentHashMap与HashMap的区别,以及它如何在并发环境中提高性能
Java面试题:请解释Java内存模型,并说明如何在多线程环境下使用synchronized关键字实现同步,阐述ConcurrentHashMap与HashMap的区别,以及它如何在并发环境中提高性能
121 0
|
安全 Java 开发者
Java多线程:synchronized关键字和ReentrantLock的区别,为什么我们可能需要使用ReentrantLock而不是synchronized?
Java多线程:synchronized关键字和ReentrantLock的区别,为什么我们可能需要使用ReentrantLock而不是synchronized?
190 0
|
安全 Java 编译器
Java多线程基础-6:线程安全问题及解决措施,synchronized关键字与volatile关键字(一)
线程安全问题是多线程编程中最典型的一类问题之一。如果多线程环境下代码运行的结果是符合我们预期的,即该结果正是在单线程环境中应该出现的结果,则说这个程序是线程安全的。 通俗来说,线程不安全指的就是某一代码在多线程环境下执行会出现bug,而在单线程环境下执行就不会。线程安全问题本质上是由于线程之间的调度顺序的不确定性,正是这样的不确定性,给我们的代码带来了很多“变数”。 本文将对Java多线程编程中,线程安全问题展开详细的讲解。
205 0
|
安全 Java 调度
Java多线程- synchronized关键字总结
Java多线程- synchronized关键字总结
99 0
|
安全 Java 数据安全/隐私保护
Java基础进阶多线程-线程安全和synchronized关键字
Java基础进阶多线程-线程安全和synchronized关键字
Java基础进阶多线程-线程安全和synchronized关键字

热门文章

最新文章