线程的等待wait(一)

一:wait()函数

  wait()是Object的里面的方法,Object是所有对象的父类,即所有对象都可以调用wait()方法。当一个线程调用一个共享变量的wait()方法时,该线程会被阻塞挂起。注意:如果调用wait()方法线程没有实现获取监视器锁,则调用wait()方法时会抛出IlleagalMonitorStateException异常

如下代码示例,线程A与线程B,在线程A中调用共享变量obj的wait()方法,在线程B中进行唤醒notify(),由于线程A,B没有获取到共享变量obj的监视器锁,故抛出lleagalMonitorStateException异常。

package org.binfa.concurrentprogramming.wait;

import lombok.extern.slf4j.Slf4j;

/**
 * Object的Wati()方法
 * @Author: jun
 * @Date: 2022/7/17 21:00
 */
@Slf4j
public class WaitTest {

    public static void main(String[] args) {

        // 定义一个共享变量
        Object obj = new Object();

        // 创建线程A
        Thread threadA = new Thread(new Runnable() {
            @Override
            public void run() {
                log.info("线程" + Thread.currentThread().getName()+"开始执行");
                try {
                    // 获取共享变量锁
                    synchronized(obj){
                        // 线程A 等待
                        log.info("线程" + Thread.currentThread().getName()+"等待");
                        obj.wait();
                    }
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }

                log.info("线程" + Thread.currentThread().getName()+"执行结束");
            }
        },"A");


        // 创建线程B
        Thread threadB = new Thread(new Runnable() {
            @Override
            public void run() {
                log.info("线程" + Thread.currentThread().getName()+"开始执行");
                // 获取共享变量锁
                synchronized (obj){
                    //  线程B 唤醒或者终端
                    log.info("线程" + Thread.currentThread().getName()+"唤醒");
                    obj.notify();
//                    threadA.interrupted();
                }
                log.info("线程" + Thread.currentThread().getName()+"执行结束");
            }
        },"B");


        // 启动线程A
        threadA.start();
        try {
            // 等待200ms,让线程B获取资源
            Thread.sleep(200);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }

        //启动线程B
        threadB.start();
    }
}


运行结果:
Exception in thread "A" java.lang.IllegalMonitorStateException
  at java.lang.Object.wait(Native Method)
线程A开始执行
  at java.lang.Object.wait(Object.java:502)
  at org.binfa.concurrentprogramming.wait.WaitTest$1.run(WaitTest.java:18)
  at java.lang.Thread.run(Thread.java:748)
main线程:main
Exception in thread "B" java.lang.IllegalMonitorStateException
线程B开始执行
  at java.lang.Object.notify(Native Method)
  at org.binfa.concurrentprogramming.wait.WaitTest$2.run(WaitTest.java:31)
  at java.lang.Thread.run(Thread.java:748)

添加synchronized(obj)监视器锁后


/**
 * Object的Wati()方法
 * @Author: jun
 * @Date: 2022/7/17 21:00
 */
public class WaitTest {

    public static void main(String[] args) {

        // 定义一个共享变量
        Object obj = new Object();

        // 创建线程A
        Thread threadA = new Thread(new Runnable() {
            @Override
            public void run() {
                System.out.println("线程" + Thread.currentThread().getName()+"开始执行");
                try {
                    // 获取共享变量锁
                    synchronized(obj){
                        // 线程A 等待
                        obj.wait();
                    }
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }

                System.out.println("线程" + Thread.currentThread().getName()+"执行结束");
            }
        },"A");


        // 创建线程B
        Thread threadB = new Thread(new Runnable() {
            @Override
            public void run() {
                System.out.println("线程" + Thread.currentThread().getName()+"开始执行");
                // 获取共享变量锁
                synchronized (obj){
                    //  线程B 唤醒或者终端
//                    obj.notify();
                    threadA.interrupted();
                }
                System.out.println("线程" + Thread.currentThread().getName()+"执行结束");
            }
        },"B");


        // 启动线程A
        threadA.start();
        try {
            // 等待200ms,让线程B获取资源
            Thread.sleep(200);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        
        //启动线程B
        threadB.start();
    }
}


执行结果:
线程A开始执行
线程B开始执行
线程B执行结束
线程A执行结束

可以看到主程序线程A启动之后,休眠了200ms让出cup执行权,线程B开始执行后调用notify()方法对阻塞线程A进行唤醒。

故:当一个线程调用一个共享变量的wait()方法时,该调用线程会被阻塞挂起,直到发生下面几件事情之一才返回:(1)其他线程调用了该共享对象的notifyO或者notifyAll)方法;(2)其他线程调用了该线程的interruptO方法,该线程抛出InterruptedException异常返回

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
Java中的线程等待wait)是一种用于线程间同步的机制。当一个线程调用对象的`wait()`方法时,它会释放该对象上的锁,并且进入等待状态,直到其他线程通过调用相同对象上的`notify()`或`notifyAll()`方法来唤醒它。 `wait()`方法必须在同步代码块或同步方法中使用,因为它要求线程先获得对象的锁才能调用。当线程调用`wait()`方法后,它会释放持有的锁,其他线程可以访问该对象并执行同步代码块或同步方法。当线程被唤醒后,它会重新尝试获得锁,并继续执行。 下面是一个简单的示例代码,演示了如何使用`wait()`和`notify()`方法实现线程等待: ```java class MyRunnable implements Runnable { private final Object lock; public MyRunnable(Object lock) { this.lock = lock; } @Override public void run() { synchronized (lock) { try { System.out.println("Thread waiting..."); lock.wait(); // 线程等待 System.out.println("Thread resumed!"); } catch (InterruptedException e) { e.printStackTrace(); } } } } public class Main { public static void main(String[] args) throws InterruptedException { Object lock = new Object(); Thread thread = new Thread(new MyRunnable(lock)); thread.start(); // 主线程等待一段时间后唤醒等待线程 Thread.sleep(2000); synchronized (lock) { lock.notify(); // 唤醒等待线程 } } } ``` 在这个示例中,线程通过调用`lock.wait()`方法进入等待状态,并且在主线程中通过调用`lock.notify()`方法将其唤醒。注意,`wait()`和`notify()`方法必须在同步代码块(这里是`synchronized (lock)`)中使用。 请注意,线程在被唤醒后,并不会立即执行,而是需要重新竞争锁资源。因此,在`notify()`方法调用之后,等待线程可能不会立即继续执行,而是需要等待其他线程释放锁才能执行。 这是Java线程等待的基本概念和用法,希望对你有所帮助!如果有更多问题,请随时提问。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值