登录
首页 >  文章 >  java教程

Java线程等待唤醒机制详解

时间:2025-09-29 11:45:56 216浏览 收藏

大家好,今天本人给大家带来文章《Java线程等待与唤醒实现方法》,文中内容主要涉及到,如果你对文章方面的知识点感兴趣,那就请各位朋友继续看下去吧~希望能真正帮到你们,谢谢!

Java中线程等待与唤醒通过wait()、notify()、notifyAll()实现,需在synchronized中调用;示例为生产者-消费者模型,使用共享资源的同步方法控制线程间通信。

如何在Java中实现线程等待与唤醒

在Java中实现线程的等待与唤醒,主要依赖于Object类提供的wait()notify()notifyAll()方法。这些方法必须在同步上下文中使用,也就是在synchronized代码块或方法中调用,否则会抛出IllegalMonitorStateException

1. wait()、notify() 和 notifyAll() 的作用

wait():使当前线程释放锁并进入等待状态,直到其他线程调用同一个对象的notify()notifyAll()方法。

notify():唤醒在此对象监视器上等待的单个线程。选择是任意的。

notifyAll():唤醒所有在此对象监视器上等待的线程。

注意:这三个方法都必须在synchronized修饰的代码块或方法中调用。

2. 基本使用示例

下面是一个简单的生产者-消费者模型,演示线程等待与唤醒机制:

class SharedResource {
    private int data;
    private boolean hasData = false;

    public synchronized void produce(int value) throws InterruptedException {
        while (hasData) {
            wait(); // 等待消费者消费
        }
        data = value;
        hasData = true;
        System.out.println("生产了: " + data);
        notify(); // 唤醒等待的消费者线程
    }

    public synchronized void consume() throws InterruptedException {
        while (!hasData) {
            wait(); // 等待生产者生产
        }
        System.out.println("消费了: " + data);
        hasData = false;
        notify(); // 唤醒等待的生产者线程
    }
}

public class WaitNotifyExample {
    public static void main(String[] args) {
        SharedResource resource = new SharedResource();

        Thread producer = new Thread(() -> {
            for (int i = 1; i  {
            for (int i = 1; i 

3. 使用条件

  • 必须使用synchronized关键字保证线程对共享资源的互斥访问。
  • 使用while而不是if判断条件,防止虚假唤醒(spurious wakeup)。
  • 每次调用notify()notifyAll()后,被唤醒的线程需要重新竞争锁。

4. 更现代的方式:使用 Lock 和 Condition

从Java 5开始,可以使用java.util.concurrent.locks.Condition实现更灵活的等待/唤醒机制。

import java.util.concurrent.locks.*;

class BetterSharedResource {
    private int data;
    private boolean hasData = false;
    private final Lock lock = new ReentrantLock();
    private final Condition hasDataCondition = lock.newCondition();
    private final Condition noDataCondition = lock.newCondition();

    public void produce(int value) throws InterruptedException {
        lock.lock();
        try {
            while (hasData) {
                noDataCondition.await(); // 等待空位
            }
            data = value;
            hasData = true;
            System.out.println("生产了: " + data);
            hasDataCondition.signal(); // 唤醒消费者
        } finally {
            lock.unlock();
        }
    }

    public void consume() throws InterruptedException {
        lock.lock();
        try {
            while (!hasData) {
                hasDataCondition.await(); // 等待数据
            }
            System.out.println("消费了: " + data);
            hasData = false;
            noDataCondition.signal(); // 唤醒生产者
        } finally {
            lock.unlock();
        }
    }
}

这种方式的优势在于一个Lock可以创建多个Condition,实现更精确的线程控制。

基本上就这些。核心是理解等待/唤醒机制背后的锁与条件判断逻辑,避免死锁和误唤醒。

以上就是《Java线程等待唤醒机制详解》的详细内容,更多关于synchronized,wait(),notify(),线程等待唤醒,LockCondition的资料请关注golang学习网公众号!

相关阅读
更多>
最新阅读
更多>
课程推荐
更多>