wait线程阻塞

介绍

线程阻塞通常是指一个线程在执行过程中由于某种原因从运行状态转为暂停状态的过程,线程阻塞会放弃CPU的使用权, 并且等待某个条件重新从暂停状态改为就绪状态。在Java中,通常使用object.wait让线程进入阻塞状态。

使用

首先我们先看wait方法,wait方法总共有三个重载方法,分别是 wait() wait(long timeout) wait(long timeout, int nanos),其中wait方法内部是调用了wait(0),顾名思义timeout就是超时时间,那么timeout等于0是什么意思呢,查看官方API如下:
这里写图片描述
意思就是如果timeout是0,不需考虑超时时间,即线程会无限等待直到被唤醒。
下面写一个demo,代码如下:

    public static void main(String args[]) {
        Thread thread = new Thread(new Runnable() {
            @Override
            public void run() {
                System.out.println( "执行线程中");
            }
        });
        thread.setName("MyThread");
        thread.start();
        
        try{
            thread.wait();
        }catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println(thread.getName() + "已经执行结束");
    }

预想的输出结果应该是

执行线程中
MyThread已经执行结束

实际上报错了

Exception in thread "main" java.lang.IllegalMonitorStateException

查看wait的API介绍有这么一句标注 The current thread must own this object’s monitor.,大意就是执行wait方法的线程必须持有线程的对象锁,使用synchronized对thread对象加锁既可,代码如下:

        try{
            synchronized (thread) {
                thread.wait();
            }
        }catch (InterruptedException e) {
            e.printStackTrace();
        }

这样一来,通过synchronize既可获取线程的对象锁,就可以达到阻塞线程的作用了,有人可能会疑惑说为什么没有调用notify方法,这是因为当线程死亡之后,相应的wait方法也会失效。

当然对于线程的阻塞,通用使用线程的join方法,代码如下:

    public static void main(String args[]) {
        Thread thread1 = new Thread(new Runnable() {
            @Override
            public void run() {
                System.out.println( "执行线程1中");
            }
        });
        thread1.setName("MyThread -- 1");
        thread1.start();

        Thread thread2 = new Thread(new Runnable() {
            @Override
            public void run() {
                System.out.println( "执行线程2中");
            }
        });
        thread2.setName("MyThread -- 2");
        thread2.start();

        try{
            thread1.join();
            thread2.join();
        }catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println(thread1.getName() + " " + thread2.getName()  + "  已经执行结束");
    }

通过调用join方法,就可以让线程阻塞,我们看一下join的方法实现:

    public final synchronized void join(long var1) throws InterruptedException {
        long var3 = System.currentTimeMillis();
        long var5 = 0L;
        if (var1 < 0L) {
            throw new IllegalArgumentException("timeout value is negative");
        } else {
            if (var1 == 0L) {
                while(this.isAlive()) {
                    this.wait(0L);
                }
            } else {
                while(this.isAlive()) {
                    long var7 = var1 - var5;
                    if (var7 <= 0L) {
                        break;
                    }

                    this.wait(var7);
                    var5 = System.currentTimeMillis() - var3;
                }
            }

        }
    }

其实就是当线程还存活时,调用wait方法。需要注意的是,join方法是synchronize的,因为需要获取对象锁。

参考

Wait方法的官方API介绍

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 在 Java 中,可以使用以下方式实现线程阻塞: 1. 通过调用 `Thread.sleep()` 方法来使线程暂停一段时间。该方法使当前线程暂停执行一段指定的时间,但不释放线程所持有的锁。 ```java try { Thread.sleep(1000); // 线程暂停1秒钟 } catch (InterruptedException e) { e.printStackTrace(); } ``` 2. 使用 `wait()` 方法来使线程进入等待状态。当一个线程执行了 `wait()` 方法后,它释放掉它所持有的锁,并且进入等待状态,直到被其它线程通过 `notify()` 或 `notifyAll()` 方法唤醒。 ```java synchronized (obj) { try { obj.wait(); // 线程进入等待状态 } catch (InterruptedException e) { e.printStackTrace(); } } ``` 3. 通过 `join()` 方法使一个线程等待另一个线程的执行完成。当一个线程执行了 `join()` 方法后,它等待指定的线程执行完成后再继续执行。 ```java Thread thread = new Thread(new Runnable() { @Override public void run() { // 线程执行的代码 } }); thread.start(); try { thread.join(); // 等待指定线程执行完成 } catch (InterruptedException e) { e.printStackTrace(); } ``` 需要注意的是,当线程进入等待状态后,它释放掉它所持有的锁,这是为了避免死锁的情况。因此,在使用 `wait()` 方法时,必须先获得锁对象的监视器锁。 ### 回答2: 在Java中,线程可以使用以下几种方法实现阻塞。 1. 使用`Thread.sleep()`方法:可以在线程调用`Thread.sleep()`方法来阻塞线程的执行一段时间。该方法接受一个以毫秒为单位的参数,指定线程要休眠的时间。调用`Thread.sleep()`方法后,线程进入阻塞状态,暂停执行指定的时间后恢复执行。 2. 使用`wait()`和`notify()`方法线程可以使用`wait()`方法进入阻塞状态,在其他线程调用相同对象的`notify()`或`notifyAll()`方法时被唤醒。`wait()`和`notify()`方法通常与`synchronized`关键字一起使用,以确保线程安全。 3. 使用`Lock`和`Condition`:Java中的`Lock`接口提供了更灵活的线程同步机制,它可以替代`synchronized`关键字来实现线程的阻塞和唤醒。通过创建一个`ReentrantLock`对象,可以使用其`newCondition()`方法创建一个`Condition`对象线程可以使用`Condition`的`await()`方法进入阻塞状态,并使用`signal()`或`signalAll()`方法通知其它线程。 4. 使用`BlockingQueue`:`BlockingQueue`是一个带有阻塞功能的队列,可以用来实现线程的阻塞和唤醒。线程可以调用`BlockingQueue`的`put()`方法将元素放入队列中,如果队列满了,线程被阻塞等待;而调用`take()`方法从队列中取出元素,如果队列为空,线程被阻塞等待。 5. 使用`Future`和`Callable`:线程可以使用`Future`和`Callable`接口来实现阻塞。`Callable`是一个带有返回值的任务,而`Future`则是对任务执行结果的引用。通过调用`Future`的`get()`方法线程可以阻塞等待任务的执行结果。 这些方法都可以在适当的地方使用,以实现Java线程的阻塞。具体使用哪种方法,取决于具体的应用场景和需求。 ### 回答3: 在Java中,线程可以通过以下几种方式实现阻塞。 1. Object类中的wait()和notify()方法:可以使用wait()方法使线程进入等待状态,直到其他线程调用相同对象上的notify()或notifyAll()方法才能被唤醒。 2. sleep()方法:通过Thread类的静态方法sleep(),可以使当前线程暂停执行一段时间。在睡眠期间,线程是阻塞的,不占用CPU时间。 3. I/O阻塞:当线程执行输入/输出操作时,如果数据未就绪,线程自动阻塞。例如,使用阻塞式的Socket读取或写入数据时,如果没有数据可用,线程将被阻塞,直到数据准备好。 4. 同步阻塞:当线程需要获取一个已经被其他线程持有的锁时,它将进入阻塞状态,直到锁被释放。这可以通过使用synchronized关键字来实现。线程可以使用synchronized关键字来获取一个对象上的锁,并在执行同步代码块时阻塞其他线程的访问。 5. 等待线程执行完毕:可以使用Thread类的join()方法,使一个线程等待另一个线程执行完毕后再继续执行。调用join()方法线程将被阻塞,直到目标线程执行完毕。 通过以上方式,可以在Java中实现线程的阻塞。阻塞可以用于控制线程的执行顺序、提高程序的稳定性和效率。但要注意适当地使用阻塞,避免造成死锁或线程饥饿等问题。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值