Java中的wait()

一、概述

wait()是Java中的一个关键字,用于线程之间的协作,线程可以调用wait()方法释放对象锁并进入等待状态,直到其他线程调用对象的notify()或notifyAll()方法来唤醒这个线程。wait()方法需要放在synchronized代码块中或者synchronized方法中,否则会抛出IllegalMonitorStateException异常。

wait()方法应该是使用线程之间的协作、互斥、同步的手段,而不是用于线程之间的通信。

二、wait()和notify()/notifyAll()的基本用法

wait()方法会将当前线程挂起,让出锁。调用wait()方法时,会依据实例对象上的锁。notify()和notifyAll()方法用于唤醒正在等待该实例对象的线程。notifyAll()会唤醒所有正在等待该实例对象的线程,而notify()会唤醒其中的一个。

public class WaitNotifyDemo {
    private final Object lock = new Object();

    private void waitForSignal() {
        synchronized (lock) {
            try {
                lock.wait();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("Thread " + Thread.currentThread().getName() + " get notified");
        }
    }

    private void notifyThread() {
        synchronized (lock) {
            lock.notify();
        }
    }

    public static void main(String[] args) throws InterruptedException {
        WaitNotifyDemo demo = new WaitNotifyDemo();

        new Thread(demo::waitForSignal, "thread1").start();

        Thread.sleep(1000);

        demo.notifyThread();
    }
}

三、wait()的超时时间

wait(long timeout)方法会在指定的超时时间内等待,如果超时还未被唤醒,线程将自动唤醒。另外,notify()或notifyAll()的调用并不需要自己持有锁,而是需要锁释放才能够发挥作用。

public class WaitNotifyTimeoutDemo {
    private final Object lock = new Object();

    private void waitForSignal(long timeout) {
        synchronized (lock) {
            try {
                lock.wait(timeout);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("Thread " + Thread.currentThread().getName() + " get notified");
        }
    }

    private void notifyThread() {
        synchronized (lock) {
            lock.notify();
        }
    }

    public static void main(String[] args) throws InterruptedException {
        WaitNotifyTimeoutDemo demo = new WaitNotifyTimeoutDemo();
        new Thread(() -> demo.waitForSignal(1000), "thread1").start();
        Thread.sleep(3000);
    }
}

四、竞争条件下的wait()

由于wait()方法必须放在synchronized代码块中,因此在多个线程同时竞争一个锁对象时,可能发生线程意外唤醒的情况。这时候需要在相应的代码中加入额外的判断语句,以防止虚假唤醒。 Java线程库提供了一个ConcurrentLock来解决这个问题,线程可以使用它来防止虚假唤醒。

public class WaitNotifyRaceConditionDemo {
    private final Object lock = new Object();
    private boolean isSignaled = false;

    private void waitForSignal() {
        synchronized (lock) {
            while(!isSignaled) {
                try {
                    lock.wait();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
            System.out.println("Thread " + Thread.currentThread().getName() + " get notified");
        }
    }

    private void notifyThread() {
        synchronized (lock) {
            isSignaled = true;
            lock.notify();
        }
    }

    public static void main(String[] args) throws InterruptedException {
        WaitNotifyRaceConditionDemo demo = new WaitNotifyRaceConditionDemo();

        new Thread(demo::waitForSignal, "thread1").start();
        Thread.sleep(1000);

        new Thread(demo::waitForSignal, "thread2").start();
        Thread.sleep(1000);

        new Thread(demo::waitForeverAndNeverBeNotified, "thread3").start();
        Thread.sleep(1000);

        demo.notifyThread();

        Thread.sleep(1000);
    }

    private static void waitForeverAndNeverBeNotified() {
        final WaitNotifyRaceConditionDemo demo = new WaitNotifyRaceConditionDemo();
        demo.waitForSignal();
    }
}

五、wait()的注意点

1、wait()方法本身可以被中断,即在wait()状态下调用interrupt()方法会是线程抛出InterruptedException异常;

2、wait()方法和synchronized代码块的使用应该遵循一个原则:wait()方法和notify()/notifyAll()方法的使用应该且只应该在对象的所有者线程(锁相关的代码操作的线程)中使用;

3、wait()和notify()/notifyAll()的语义需要开发者精细把控,例如代码中WaitNotifyRaceConditionDemo示例中,在notifyThread()方法中,需要先将notify()之后再修改状态。

六、结论

wait()是Java中的一个重要关键字,用于线程之间的协作,线程可以调用wait()方法释放对象锁并进入等待状态,直到其他线程调用对象的notify()或notifyAll()方法来唤醒该线程。wait()方法需要放在synchronized代码块中或者synchronized方法中,否则会抛出IllegalMonitorStateException异常。wait()方法应该是使用线程之间的协作、互斥、同步的手段,而不是用于线程之间的通信。在使用wait()方法和notify()/notifyAll()方法的时候需要注意竞争条件,需要额外加判断以及确保wait()和notify()/notifyAll()语义的正确性。

原创文章,作者:小蓝,如若转载,请注明出处:https://www.506064.com/n/301038.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
小蓝小蓝
上一篇 2024-12-29 14:19
下一篇 2024-12-29 14:19

相关推荐

  • Java JsonPath 效率优化指南

    本篇文章将深入探讨Java JsonPath的效率问题,并提供一些优化方案。 一、JsonPath 简介 JsonPath是一个可用于从JSON数据中获取信息的库。它提供了一种DS…

    编程 2025-04-29
  • java client.getacsresponse 编译报错解决方法

    java client.getacsresponse 编译报错是Java编程过程中常见的错误,常见的原因是代码的语法错误、类库依赖问题和编译环境的配置问题。下面将从多个方面进行分析…

    编程 2025-04-29
  • Java腾讯云音视频对接

    本文旨在从多个方面详细阐述Java腾讯云音视频对接,提供完整的代码示例。 一、腾讯云音视频介绍 腾讯云音视频服务(Cloud Tencent Real-Time Communica…

    编程 2025-04-29
  • Java Bean加载过程

    Java Bean加载过程涉及到类加载器、反射机制和Java虚拟机的执行过程。在本文中,将从这三个方面详细阐述Java Bean加载的过程。 一、类加载器 类加载器是Java虚拟机…

    编程 2025-04-29
  • Java Milvus SearchParam withoutFields用法介绍

    本文将详细介绍Java Milvus SearchParam withoutFields的相关知识和用法。 一、什么是Java Milvus SearchParam without…

    编程 2025-04-29
  • Java 8中某一周的周一

    Java 8是Java语言中的一个版本,于2014年3月18日发布。本文将从多个方面对Java 8中某一周的周一进行详细的阐述。 一、数组处理 Java 8新特性之一是Stream…

    编程 2025-04-29
  • Java判断字符串是否存在多个

    本文将从以下几个方面详细阐述如何使用Java判断一个字符串中是否存在多个指定字符: 一、字符串遍历 字符串是Java编程中非常重要的一种数据类型。要判断字符串中是否存在多个指定字符…

    编程 2025-04-29
  • VSCode为什么无法运行Java

    解答:VSCode无法运行Java是因为默认情况下,VSCode并没有集成Java运行环境,需要手动添加Java运行环境或安装相关插件才能实现Java代码的编写、调试和运行。 一、…

    编程 2025-04-29
  • Java任务下发回滚系统的设计与实现

    本文将介绍一个Java任务下发回滚系统的设计与实现。该系统可以用于执行复杂的任务,包括可回滚的任务,及时恢复任务失败前的状态。系统使用Java语言进行开发,可以支持多种类型的任务。…

    编程 2025-04-29
  • Java 8 Group By 会影响排序吗?

    是的,Java 8中的Group By会对排序产生影响。本文将从多个方面探讨Group By对排序的影响。 一、Group By的概述 Group By是SQL中的一种常见操作,它…

    编程 2025-04-29

发表回复

登录后才能评论