第二十三讲 多线程——sleep( )和wait( )的5个区别

版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/yerenyuan_pku/article/details/82832548

sleep(休眠)和wait(等待)方法是Java多线程中常用的两个方法,它们有什么区别及一些该注意的地方有哪些呢?下面给大家一一讲解。

区别1:使用限制

使用sleep()方法可以让当前线程休眠,时间一到当前线程继续往下执行,在任何地方都能使用,但需要捕获InterruptedException异常。

try {
    Thread.sleep(3000L);
} catch (InterruptedException e) {
    e.printStackTrace();
}

而使用wait()方法则必须放在synchronized块里面,同样需要捕获InterruptedException异常,并且需要获取对象的锁。

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

而且wait需要额外的方法notify()/notifyAll()进行唤醒,它们同样需要放在synchronized块里面,且需要获取对象的锁。

synchronized (lock) {
    // 随机唤醒
    lock.notify();

    // 唤醒全部
    lock.notifyAll();
}

当然也可以使用带时间的wait(long millis)方法,时间一到,无需其他线程唤醒,也会重新竞争获取对象的锁继续执行。

区别2:使用场景

sleep()一般用于当前线程休眠,或者轮循暂停操作,wait()则多用于多线程之间的通信。

区别3:所属类

sleep()是Thread类的静态本地方法,wait()则是Object类的本地方法。

java.lang.Thread#sleep

public static native void sleep(long millis) throws InterruptedException;

java.lang.Object#wait

public final native void wait(long timeout) throws InterruptedException;

为什么要这样设计呢?
因为sleep()让当前线程休眠,不涉及到对象类,也不需要获得对象的锁,所以是线程类的方法;wait()是让获得对象锁的线程实现等待,前提是要先获得对象的锁,所以是类的方法。

区别4:释放锁

Object lock = new Object();
synchronized (lock) {
    try {
        lock.wait(3000L);
        Thread.sleep(2000L);
    } catch (InterruptedException e) {
        e.printStackTrace();
    }
}

如上代码所示,wait()可以释放当前线程对lock对象锁的持有,而sleep()则不会。简而言之,就是:

wait():释放CPU资源,同时释放锁;
sleep():释放CPU资源,不释放锁。

区别5:线程切换

sleep()会让出CPU执行时间且强制上下文切换,而wait()则不一定,wait()后可能还是有机会重新竞争到锁继续执行的。

猜你喜欢

转载自blog.csdn.net/yerenyuan_pku/article/details/82832548