JAVA线程sleep()和wait()详解及实例
JAVA线程sleep()和wait()详解及实例
sleep
1.sleep是Thread的一个静态(static)方法。使得Runnable实现的线程也可以使用sleep方法。而且避免了线程之前相互调用sleep()方法,引发死锁。
2.sleep()执行时需要赋予一个沉睡时间。在沉睡期间(阻塞线程期间),CPU会放弃这个线程,执行其他任务。当沉睡时间到了之后,该线程会自动苏醒,不过此时线程不会立刻被执行,而是要等CPU分配资源,和其他线程进行竞争。
3.此外如果这个线程之前获取了一个机锁,在沉睡期间,这个机锁不会释放。其他等待这个机锁的程序,必须等待这个线程醒来,且执行完后才能运行。
sleep相关代码
publicclassThreadTest2{ publicstaticvoidmain(String[]args){ System.out.println("beginourtest"); ThreadSleepsleep=newThreadSleep(); try{ Threadthread1=newThread(sleep,"路人甲"); Threadthread2=newThread(sleep,"路人乙"); thread1.start(); thread2.start(); }catch(Exceptione){ e.printStackTrace(); } System.out.println("testisover"); } } classThreadSleepimplementsRunnable{ intcount=0; @Override publicvoidrun(){ System.out.println(Thread.currentThread().getName()+"say:hellosleep!!"); count(); } publicvoidcount(){ while(count<20){ System.out.println(Thread.currentThread().getName()+"say:countis"+count); try{ count++; Thread.sleep(100); }catch(Exceptione){ e.printStackTrace(); } } } }
输出日志
beginourtest testisover 路人甲say:hellosleep!! 路人甲say:countis0 路人乙say:hellosleep!! 路人乙say:countis1 路人甲say:countis2 路人乙say:countis2 路人甲say:countis4 路人乙say:countis4 路人甲say:countis6 路人乙say:countis7 路人乙say:countis8 路人甲say:countis8 路人甲say:countis10 路人乙say:countis10 路人乙say:countis12 路人甲say:countis12 路人乙say:countis14 路人甲say:countis14 路人甲say:countis16 路人乙say:countis16 路人甲say:countis18 路人乙say:countis18
通过日志可以发现线程甲和线程乙基本是交替执行,但是并不规律,且出现了并发问题。
该情况是由于代码中设置了睡眠时间为100毫秒,由于count递增执行速度很快,所以线程差不多是同时睡眠,然后同时苏醒并导致了并发的出现。
接下来要添加synchronize块,检查sleep时机锁是否释放
publicclassThreadTest2{ publicstaticvoidmain(String[]args){ System.out.println("beginourtest"); ThreadSleepsleep=newThreadSleep(); try{ Threadthread1=newThread(sleep,"路人甲"); Threadthread2=newThread(sleep,"路人乙"); thread1.start(); thread2.start(); }catch(Exceptione){ e.printStackTrace(); } System.out.println("testisover"); } } classThreadSleepimplementsRunnable{ intcount=0; @Override publicvoidrun(){ System.out.println(Thread.currentThread().getName()+"say:hellosleep!!"); count(); } publicvoidcount(){ while(count<20){ synchronized(this){ System.out.println(Thread.currentThread().getName()+"say:countis"+count); try{ count++; Thread.sleep(100); }catch(Exceptione){ e.printStackTrace(); } } } } }
输出日志
beginourtest 路人甲say:hellosleep!! 路人甲say:countis0 testisover 路人乙say:hellosleep!! 路人甲say:countis1 路人甲say:countis2 路人甲say:countis3 路人甲say:countis4 路人甲say:countis5 路人甲say:countis6 路人甲say:countis7 路人甲say:countis8 路人甲say:countis9 路人甲say:countis10 路人甲say:countis11 路人甲say:countis12 路人甲say:countis13 路人甲say:countis14 路人甲say:countis15 路人甲say:countis16 路人甲say:countis17 路人甲say:countis18 路人甲say:countis19 路人乙say:countis20
通过日志可以看出,基本是线程甲在执行,这是因为sleep时,机锁一直在线程甲上,所以线程乙只能一直等待直到线程甲释放锁。
wait
1.wait()是Object类的一个方法。当调用wait()方法时,该线程会进入和该对象相关的等待池中,并释放它所拥有的机锁。
2.执行wait()后,必须使用notify()方法或notifyAll()方法或设置等待时间(wait(longtime))唤醒在等待线程池中的线程。
3.wait()必须放在synchronizedblock中,否则会在运行时报“java.lang.IllegalMonitorStateException”异常
wait相关代码
publicclassThreadTest2{ publicstaticvoidmain(String[]args){ System.out.println("beginourtest"); ThreadSleepsleep=newThreadSleep(); try{ Threadthread1=newThread(sleep,"路人甲"); Threadthread2=newThread(sleep,"路人乙"); thread1.start(); thread2.start(); }catch(Exceptione){ e.printStackTrace(); } System.out.println("testisover"); } } classThreadSleepimplementsRunnable{ intcount=0; @Override publicvoidrun(){ System.out.println(Thread.currentThread().getName()+"say:hellosleep!!"); count(); } publicvoidcount(){ while(count<20){ synchronized(this){ System.out.println(Thread.currentThread().getName()+"say:countis"+count); try{ count++; this.wait(100); }catch(Exceptione){ e.printStackTrace(); } } } } }
输出日志
beginourtest 路人甲say:hellosleep!! 路人甲say:countis0 testisover 路人乙say:hellosleep!! 路人乙say:countis1 路人甲say:countis2 路人乙say:countis3 路人甲say:countis4 路人乙say:countis5 路人甲say:countis6 路人乙say:countis7 路人甲say:countis8 路人乙say:countis9 路人甲say:countis10 路人乙say:countis11 路人甲say:countis12 路人乙say:countis13 路人乙say:countis14 路人甲say:countis15 路人乙say:countis16 路人甲say:countis17 路人乙say:countis18 路人甲say:countis19
通过日志可以发现在wait的情况下,机锁会被释放。
感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!