JAVA线程sleep和wait方法区别

一、

sleep 是线程类(Thread)的方法,导致此线程暂停执行指定时间,给执行机会给其他线程,但是监控状态依然保持,到时后会自动恢复,调用sleep 不会释放对象锁。由于没有释放对象锁,所以不能调用里面的同步方法。

sleep()使当前线程进入停滞状态(阻塞当前线程),让出CUP的使用、目的是不让当前线程独自霸占该进程所获的CPU资源,以留一定时间给其他线程执行的机会;
sleep()是Thread类的Static(静态)的方法;因此他不能改变对象的机锁,所以当在一个Synchronized块中调用Sleep()方法是,线程虽然休眠了,但是对象的机锁并木有被释放,其他线程无法访问这个对象(即使睡着也持有对象锁)。
在sleep()休眠时间期满后,该线程不一定会立即执行,这是因为其它线程可能正在运行而且没有被调度为放弃执行,除非此线程具有更高的优先级。

wait()方法是Object类里的方法;当一个线程执行到wait()方法时,它就进入到一个和该对象相关的等待池中,同时失去(释放)了对象的机锁(暂时失去机锁,wait(long timeout)超时时间到后还需要返还对象锁);可以调用里面的同步方法,其他线程可以访问;
wait()使用notify或者notifyAlll或者指定睡眠时间来唤醒当前等待池中的线程。
wiat()必须放在synchronized block中,否则会在program runtime时扔出”java.lang.IllegalMonitorStateException“异常。

二、

sleep必须捕获异常,而wait,notify和notifyAll不需要捕获异常

sleep方法属于Thread类中方法,表示让一个线程进入睡眠状态,等待一定的时间之后,自动醒来进入到可运行状态,不会马上进入运行状态,因为线程调度机制恢复线程的运行也需要时间,一个线程对象调用了sleep方法之后,并不会释放他所持有的所有对象锁,所以也就不会影响其他进程对象的运行。但在sleep的过程中过程中有可能被其他对象调用它的interrupt(),产生InterruptedException异常,如果你的程序不捕获这个异常,线程就会异常终止,进入TERMINATED状态,如果你的程序捕获了这个异常,那么程序就会继续执行catch语句块(可能还有finally语句块)以及以后的代码。

注意sleep()方法是一个静态方法,也就是说他只对当前对象有效,通过t.sleep()让t对象进入sleep,这样的做法是错误的,它只会是使当前线程被sleep 而不是t线程

wait属于Object的成员方法,一旦一个对象调用了wait方法,必须要采用notify()和notifyAll()方法唤醒该进程;如果线程拥有某个或某些对象的同步锁,那么在调用了wait()后,这个线程就会释放它持有的所有同步资源,而不限于这个被调用了wait()方法的对象。wait()方法也同样会在wait的过程中有可能被其他对象调用interrupt()方法而产生

三 、

这两者的施加者是有本质区别的. 
sleep()是让某个线程暂停运行一段时间,其控制范围是由当前线程决定,也就是说,在线程里面决定.好比如说,我要做的事情是 "点火->烧水->煮面",而当我点完火之后我不立即烧水,我要休息一段时间再烧.对于运行的主动权是由我的流程来控制.

而wait(),首先,这是由某个确定的对象来调用的,将这个对象理解成一个传话的人,当这个人在某个线程里面说"暂停!",也是 thisOBJ.wait(),这里的暂停是阻塞,还是"点火->烧水->煮饭",thisOBJ就好比一个监督我的人站在我旁边,本来该线 程应该执行1后执行2,再执行3,而在2处被那个对象喊暂停,那么我就会一直等在这里而不执行3,但正个流程并没有结束,我一直想去煮饭,但还没被允许, 直到那个对象在某个地方说"通知暂停的线程启动!",也就是thisOBJ.notify()的时候,那么我就可以煮饭了,这个被暂停的线程就会从暂停处 继续执行.


其实两者都可以让线程暂停一段时间,但是本质的区别是一个线程的运行状态控制,一个是线程之间的通讯的问题

 在java.lang.Thread类中,提供了sleep(),
而java.lang.Object类中提供了wait(), notify()和notifyAll()方法来操作线程
sleep()可以将一个线程睡眠,参数可以指定一个时间。
而wait()可以将一个线程挂起,直到超时或者该线程被唤醒。
    wait有两种形式wait()和wait(milliseconds).
sleep和wait的区别有:
  1,这两个方法来自不同的类分别是Thread和Object
  2,最主要是sleep方法没有释放锁,而wait方法释放了锁,使得其他线程可以使用同步控制块或者方法。
  3,wait,notify和notifyAll只能在同步控制方法或者同步控制块里面使用,而sleep可以在
    任何地方使用
   synchronized(x){
      x.notify()
     //或者wait()
   }
   4,sleep必须捕获异常,而wait,notify和notifyAll不需要捕获异常

四、下面用生产者和消费者举例

复制代码
 1 package ProducterOrCustomer;
 2 
 3 public class ThreadDemo {
 4 
 5     public static void main(String[] args) {
 6 
 7         Food food = new Food();
 8 
 9         Thread thread1 = new Thread(new Customer(food));  //消费者线程
10         Thread thread2 = new Thread(new Producter(food)); //生产者线程
11 
12         thread1.start();
13         thread2.start();
14     }
15 
16 }
复制代码
复制代码
 1 package ProducterOrCustomer;
 2 
 3 //  消费者
 4 public class Customer implements Runnable{
 5 
 6     private Food food;
 7 
 8     public Customer(Food food){
 9         this.food = food;
10     }
11 
12     @Override
13     public void run(){
14         for (int i = 0; i < 3; i++){
15             food.get();
16         }
17     }
18 }
复制代码
复制代码
 1 package ProducterOrCustomer;
 2 
 3 // 生产者
 4 public class Producter implements Runnable{
 5 
 6     private Food food;
 7 
 8     public Producter(Food food){
 9         this.food = food;
10     }
11 
12     @Override
13     public void run(){
14         for (int i = 0; i < 2; i++){
15             if (i%2 == 0){
16                 food.set("红烧肉","好吃");
17             }else {
18                 food.set("八宝鸭","都是米饭没鸭");
19             }
20         }
21     }
22 }
复制代码
复制代码
 1 package ProducterOrCustomer;
 2 
 3 public class Food {
 4 
 5     private String foodName;
 6 
 7     private String foodTaste;
 8 
 9     private Boolean flag = true;
10 
11 
12     /** 初始值flag = true时,set()所在的线程处于正常执行状态,get()所在的线程处于wait()状态被挂起,
13      *  当执行到set()方法的34行时,flag = false,get()所造的线程被唤醒,这时,set()所造的线程处于wait()状态被挂起,
14      *  依次循环,所以每次都是先生产再消费,这就是生产者与消费者模式
15      */
16 
17     //生产
18     public synchronized  void set(String foodName,String foodTaste){
19         if (!flag ){
20             try {
21                 this.wait();  //当前线程进入等待状态,并让出CPU,并且释放监视器的锁
22             }catch (InterruptedException e){
23                 e.printStackTrace();
24             }
25         }
26         System.out.println("生产 " + foodName + " " + foodTaste);
27         this.setFoodName(foodName);
28         this.setFoodTaste(foodTaste);
29         try {
30             Thread.sleep(500);  //当前线程进入等待状态,让出cpu,不释放锁
31         }catch (InterruptedException e){
32             e.printStackTrace();
33         }
34         flag = false;
35         this.notify();
36     }
37 
38     //消费
39     public synchronized void get(){
40         if (flag){
41             try {
42                 this.wait();
43             }catch (InterruptedException e){
44                 e.printStackTrace();
45             }
46         }
47         System.out.println("消费"+this.foodName + " " + this.foodTaste);
48 
49         try {
50             Thread.sleep(500);
51         }catch (InterruptedException e){
52             e.printStackTrace();
53         }
54         flag = true;
55         this.notify();
56     }
57 
58     public Food(){
59 
60     }
61 
62     public Food(String foodName,String foodTaste){
63         this.foodName = foodName;
64         this.foodTaste = foodTaste;
65     }
66 
67     public String getFoodName() {
68         return foodName;
69     }
70 
71     public void setFoodName(String foodName) {
72         this.foodName = foodName;
73     }
74 
75     public String getFoodTaste() {
76         return foodTaste;
77     }
78 
79     public void setFoodTaste(String foodTaste) {
80         this.foodTaste = foodTaste;
81     }
82 
83     public Boolean getFlag() {
84         return flag;
85     }
86 
87     public void setFlag(Boolean flag) {
88         this.flag = flag;
89     }
90 }
复制代码

 

posted @ 2023-02-14 13:58  阿风小子  阅读(160)  评论(0编辑  收藏  举报