JAVA和Nginx 教程大全

网站首页 > 精选教程 正文

详细介绍一下Java中的wait()方法与sleep()方法的区别与联系?

wys521 2024-10-29 16:59:43 精选教程 23 ℃ 0 评论

wait()和sleep()方法都是在java中用来控制线程执行和暂停的方法,也是在面试过程中被问到最多的两个方法,下面我们就来详细介绍一下两个方法之间的主要的区别与联系。

定义与用途

wait()方法是一个对象方法,是属于java.lang.Object类的方法,sleep()则是一个静态方法,是一个属于java.lang.Thread类的方法。

wait()方法的主要作用是用于线程间的通信,一般情况下与notify()或notifyAll()方法一起使用,当一个线程调用wait()方法时,它会释放当前持有的对象锁,并进入等待状态,直到其他线程调用同一对象的notify()或notifyAll()方法,线程重新尝试获取锁。注意这里是尝试获取而不是一定就能得到。

sleep()方法的主要作用是用于暂停当前线程的执行,与wait()方法不同的是,调用 sleep() 方法后,线程会进入休眠状态,但不释放任何锁,直到指定的时间到期,之后线程会自动恢复执行。

线程状态

当线程调用了wait()方法的时候,它的状态会变为WAITING状态,直到它收到通知或者被中断。而当线程调用sleep()方法的时候,它的状态会变为TIMED_WAITING状态,意思是在指定时间内会保持此状态,直到时间到了继续执行。

锁的释放

调用wait() 方法会释放当前对象的监视器锁,也就是我们常说的对象锁,这个时候就允许其他线程访问该对象的同步方法或代码块。而在调用了sleep() 方法之后,线程不会释放任何锁,线程仍然持有其锁定的对象。其他线程无法进入同步代码块,直到当前线程醒来。

需要的上下文

调用wait()方法,是只能在同步环境中调用,也就是说必须在一个被synchronized关键字修饰的方法或代码块中调用,否则会抛出IllegalMonitorStateException。但是对于sleep()方法的调用来讲,它可以在任何上下文中调用,不需要持有对象锁。可以在同步代码块中或非同步代码块中使用。

参数

wait(long timeout)可以指定线程等待的时间,但是如果在该时间内没有被通知,线程就会自动返回。对于sleep(long millis)方法来讲,它也会接收一个毫秒值来指定线程休眠的时间,当然我们也可以使用sleep(long millis, int nanos)来精确指定休眠的时间。

使用示例

wait()方法示例

public class WaitNotifyExample {
    private static final Object lock = new Object();
    
    public static void main(String[] args) {
        Thread waiter = new Thread(() -> {
            synchronized (lock) {
                try {
                    System.out.println("Waiting for notification...");
                    lock.wait(); // 进入等待状态
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println("Notified and resumed.");
            }
        });

        Thread notifier = new Thread(() -> {
            try {
                Thread.sleep(2000); // 等待2秒后通知
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            synchronized (lock) {
                System.out.println("Sending notification...");
                lock.notify(); // 发送通知
            }
        });

        waiter.start();
        notifier.start();
    }
}

sleep() 方法示例

public class SleepExample {
    public static void main(String[] args) {
        System.out.println("Thread is going to sleep...");
        try {
            Thread.sleep(2000); // 休眠2秒
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println("Thread has woken up.");
    }
}

总结

wait()和sleep()两个方法都可以使线程进入等待状态,但它们的目的和使用方式不同。其中wait()方法主要用于线程间的协作和通信,而sleep()方法则主要用于简单地延迟线程的执行。理解这两者的区别与联系对于编写高效的多线程程序至关重要。

Tags:

本文暂时没有评论,来添加一个吧(●'◡'●)

欢迎 发表评论:

最近发表
标签列表