登录
首页 >  文章 >  java教程

Java锁与条件实现等待唤醒机制

时间:2026-05-13 20:31:32 147浏览 收藏

Java中的Lock与Condition组合提供了比synchronized更强大、灵活的线程同步机制,尤其擅长实现精准的等待唤醒协作——如生产者消费者模型中通过notFull和notEmpty两个独立条件实现双向、定向唤醒,避免了传统wait/notify的“虚假唤醒”和单条件限制;它支持多条件绑定、公平锁策略及中断响应,显著提升了并发编程的可控性与健壮性,但需严格遵循加锁-等待-唤醒-解锁的规范流程,是高阶Java并发开发不可或缺的核心技能。

Java如何使用Lock与Condition实现等待唤醒机制_Java同步锁的进阶使用

在Java多线程编程中,LockCondition 提供了比synchronized更灵活的线程同步控制方式。通过它们可以实现精确的等待唤醒机制,尤其适用于复杂的线程协作场景。

Lock与Condition的基本概念

Lock接口是synchronized的增强替代方案,支持更细粒度的锁控制。而Condition则类似于Object的wait()和notify(),但功能更强大,允许一个Lock绑定多个Condition实例。

使用步骤如下:

  • 创建ReentrantLock实例作为锁
  • 调用lock.newCondition()获取Condition对象
  • 线程在条件不满足时调用condition.await()进入等待状态
  • 其他线程改变状态后调用condition.signal()或signalAll()唤醒等待线程

实现生产者消费者模型

下面是一个典型的生产者消费者示例,展示如何用Lock+Condition协调两个线程:

import java.util.LinkedList;
import java.util.concurrent.locks.*;

public class ProducerConsumer {
    private final LinkedList<String> buffer = new LinkedList<>();
    private final int MAX_SIZE = 5;
    private final Lock lock = new ReentrantLock();
    private final Condition notFull = lock.newCondition();
    private final Condition notEmpty = lock.newCondition();

    public void produce(String data) throws InterruptedException {
        lock.lock();
        try {
            while (buffer.size() == MAX_SIZE) {
                System.out.println("缓冲区满,生产者等待...");
                notFull.await();
            }
            buffer.add(data);
            System.out.println("生产:" + data);
            notEmpty.signal(); // 唤醒消费者
        } finally {
            lock.unlock();
        }
    }

    public String consume() throws InterruptedException {
        lock.lock();
        try {
            while (buffer.isEmpty()) {
                System.out.println("缓冲区空,消费者等待...");
                notEmpty.await();
            }
            String data = buffer.removeFirst();
            System.out.println("消费:" + data);
            notFull.signal(); // 唤醒生产者
            return data;
        } finally {
            lock.unlock();
        }
    }
}

这个例子中,notFull用于生产者等待,notEmpty用于消费者等待,实现了双向通信。

Condition的优势与注意事项

相比传统的synchronized+wait/notify,Condition有以下优势:

  • 一个锁可绑定多个Condition,实现精准唤醒
  • 支持公平锁模式,避免线程饥饿
  • await/signal操作必须在lock()/unlock()之间执行
  • 支持中断响应(await()可被interrupt打断)

使用时要注意:必须始终在try-finally块中释放锁,防止死锁;signal()只唤醒一个线程,signalAll()唤醒所有等待线程。

基本上就这些。掌握Lock与Condition的配合使用,能让你更好地控制复杂并发场景下的线程协作。

文中关于java,lock的知识介绍,希望对你的学习有所帮助!若是受益匪浅,那就动动鼠标收藏这篇《Java锁与条件实现等待唤醒机制》文章吧,也可关注golang学习网公众号了解相关技术文章。

资料下载
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>