A股上市公司传智教育(股票代码 003032)旗下技术交流社区北京昌平校区

 找回密码
 加入黑马

QQ登录

只需一步,快速开始

© Dej@vu 中级黑马   /  2013-8-6 18:10  /  1642 人查看  /  7 人回复  /   0 人收藏 转载请遵从CC协议 禁止商业使用本文

在什么情况下用他们 有什么区别?求帮助

评分

参与人数 1黑马币 +3 收起 理由
神之梦 + 3 神马都是浮云

查看全部评分

7 个回复

正序浏览
网上的答案:sleep是线程类(Thread)的方法,导致此线程暂停执行指定时间,给执行机会给其他线程,但是监控状态依然保持,到时后会自动恢复。调用sleep不会释放对象锁。 wait是Object类的方法,对此对象调用wait方法导致本线程放弃对象锁,进入等待此对象的等待锁定池,只有针对此对象发出notify方法(或notifyAll)后本线程才进入对象锁定池准备获得对象锁进入运行状态。)

sleep就是正在执行的线程主动让出cpu,cpu去执行其他线程,在sleep指定的时间过后,cpu才会回到这个线程上继续往下执行,如果当前线程进入了同步锁,sleep方法并不会释放锁,即使当前线程使用sleep方法让出了cpu,但其他被同步锁挡住了的线程也无法得到执行。wait是指在一个已经进入了同步锁的线程内,让自己暂时让出同步锁,以便其他正在等待此锁的线程可以得到同步锁并运行,只有其他线程调用了notify方法(notify并不释放锁,只是告诉调用过wait方法的线程可以去参与获得锁的竞争了,但不是马上得到锁,因为锁还在别人手里,别人还没释放。如果notify方法后面的代码还有很多,需要这些代码执行完后才会释放锁,可以在notfiy方法后增加一个等待和一些代码,看看效果),调用wait方法的线程就会解除wait状态和程序可以再次得到锁后继续向下运行。对于wait的讲解一定要配合例子代码来说明,才显得自己真明白。
package com.huawei.interview;

public class MultiThread {

        /**
         * @param args
         */
        public static void main(String[] args) {
                // TODO Auto-generated method stub
                new Thread(new Thread1()).start();
                try {
                        Thread.sleep(10);
                } catch (InterruptedException e) {
                        // TODO Auto-generated catch block
                        e.printStackTrace();
                }
                new Thread(new Thread2()).start();               
        }
       
       
        private static class Thread1 implements Runnable
        {

                @Override
                public void run() {
                        // TODO Auto-generated method stub
//由于这里的Thread1和下面的Thread2内部run方法要用同一对象作为监视器,我们这里不能用this,因为在Thread2里面的this和这个Thread1的this不是同一个对象。我们用MultiThread.class这个字节码对象,当前虚拟机里引用这个变量时,指向的都是同一个对象。
                        synchronized (MultiThread.class) {

                                System.out.println("enter thread1...");
                               
                                System.out.println("thread1 is waiting");
                                try {
                        //释放锁有两种方式,第一种方式是程序自然离开监视器的范围,也就是离开了synchronized关键字管辖的代码范围,另一种方式就是在synchronized关键字管辖的代码内部调用监视器对象的wait方法。这里,使用wait方法释放锁。
                                        MultiThread.class.wait();
                                } catch (InterruptedException e) {
                                        // TODO Auto-generated catch block
                                        e.printStackTrace();
                                }
                               
                                System.out.println("thread1 is going on...");
                                System.out.println("thread1 is being over!");                       
                        }
                }
               
        }
       
        private static class Thread2 implements Runnable
        {

                @Override
                public void run() {
                        // TODO Auto-generated method stub
                        synchronized (MultiThread.class) {
                       
                                System.out.println("enter thread2...");
                               
                                System.out.println("thread2 notify other thread can release wait status..");
//由于notify方法并不释放锁, 即使thread2调用下面的sleep方法休息了10毫秒,但thread1仍然不会执行,因为thread2没有释放锁,所以Thread1无法得不到锁。

                                MultiThread.class.notify();
                               
                                System.out.println("thread2 is sleeping ten millisecond...");
                                try {
                                        Thread.sleep(10);
                                } catch (InterruptedException e) {
                                        // TODO Auto-generated catch block
                                        e.printStackTrace();
                                }
                               
                                System.out.println("thread2 is going on...");
                                System.out.println("thread2 is being over!");
                               
                        }
                }
               
        }       

}
回复 使用道具 举报
本帖最后由 封号了... 于 2013-8-7 18:53 编辑

虽然都有停得意思,不过sleep()是暂停,就想定了一个每隔一段时间便会响铃的闹钟一样,那么每隔一段时间其间代码又会运行起来,就这样反复执行。
而wait()是停止的意思,如果没有得到外界的通知notify(),自己是不能擅自运行的,就像一个人被点了穴,需要外界给他解开穴道notify(),他才能继续运行,
不过一旦运行如若没有干扰是不会自行停下来的!
回复 使用道具 举报
在多线程中,sleep()可以给它进行设置睡眠的时间当时间一到线程就会解冻,而wait()使线程冻结必须使用notify()才能使线程运行。
回复 使用道具 举报
本帖最后由 李健09 于 2013-8-8 00:25 编辑

我是这样想的
sleep在进程当中是让正在执行的这个线程停止一会休息一下而后继续执行这个线程下面的语句,是这个线程没有执行完整并没有放弃执行权,
wait在进程当中让这个线程休息下停止对资源执行操作;由别的(其他的线程)线程对该资源执行操作,等notify()把正在休息的线程叫醒他才对该资源执行操作,
例如有一资源a,有两个线程b和c,他们都对资源a在执行操作,b在操作a 的时候,c线程使用wait那么此时c就不可以操作a直到等notify()把正在休息c的线程叫醒他才对该资源a执行操作
这个已经放弃了该资源的执行权
回复 使用道具 举报
sleep()和wait()的区别你可以这么理解:
都是叫线程睡觉去了,睡觉期间他会失去执行权。但是sleep()给线程定了闹钟,过了一段时间他自动就醒过来了。但是wait()没有闹钟,必须要通过notify()主动把线程叫醒。
回复 使用道具 举报
本帖最后由 施大勇 于 2013-8-6 21:12 编辑

sleep()在老版本java中可以应用到run()主体中进行定时控制执行的作用,过一段时间执行一次。调用它的线程自己到时会自动解除这种状态。在新版本中可以用线程池 ScheduledExecutorServer来控制定时调度了。
wait()调用它的线程自己不能解除这种状态,需要别的线程来notify()或notifyAll()来唤醒它。新版本中它的功能由Condition类的signal()和signalAll()来替换了。


wait()一般来说,有多个线程操作同一资源,但不希望一直由一个线程操作资源多次,而是线程间交互进行操作,即你一次,我一次时用。像生产者,消费者的问题就用它。

而sleep()应该是想在一段时间之后再调用某些代码时用,比如我想10秒间隔产生一个随机数,产生一百个。for ( 这里是条件,100次){ sleep(); 产生数字}

评分

参与人数 1技术分 +1 收起 理由
神之梦 + 1 赞一个!

查看全部评分

回复 使用道具 举报
sleep()它没有释放系统资源,等到限定的时间到了之后,调用它的进程自动被唤醒。
而wait()会释放系统资源,使得其他线程可以使用同步控制块或者方法。

评分

参与人数 1技术分 +1 收起 理由
神之梦 + 1 赞一个!

查看全部评分

回复 使用道具 举报
您需要登录后才可以回帖 登录 | 加入黑马