Thread-方法以及wait、notify简介

Thread.sleep()
1.静态方法是定义在Thread类中。
2.Thread.sleep()方法用来暂停当前执行的线程,将CPU使用权释放给线程调度器,但不释放锁(也就是说如果有synchronized同步快,其他线程仍然不能访问共享数据),进入sleeping状态
的线程不能被notify方法唤醒。在自动苏醒(超过指定的毫秒数)之后不保证能获取到CPU,它会先进入就绪态,与其他线程(不分高低优先级)竞争CPU。有两个sleep方法,一个只有一个毫
秒参数,另一个有毫秒和纳秒两个参数。如果其他的线程中断了一个休眠的线程,sleep方法会抛出Interrupted Exception。sleep是帮助其他线程获得运行机会的最好方法,但不会让出锁。

Thread.yield()
1.静态方法是定义在Thread类中。
2.yield方法会临时暂停当前正在执行的线程,来让有同样优先级的正在等待的线程有机会执行。如果没有正在等待的线程,或者所有正在等待的线程的优先级都比较低,那么该线程会继续运
行(只能使同优先级的线程有执行的机会)。执行了yield方法的线程什么时候会继续运行由线程调度器来决定,不同的厂商可能有不同的行为。yield方法不保证当前的线程会暂停或者停止
,但是可以保证当前线程在调用yield方法时会放弃CPU和锁。

join()实例方法
1.join()方法使调用该方法的线程在此之前执行完毕,也就是等待该方法的线程执行完毕后再往下继续执行。注意该方法也需要捕捉异常。

wait
1.定义在Object类中的非静态方法。

2.用于线程间通信的,会释放cpu和锁。应在同步环境中调用。能够被notify和notifyAll唤醒。
3.void wait() 在其他线程调用此对象的 notify() 方法或者 notifyAll()方法前,导致当前线程等待。

void wait(long timeout)在其他线程调用此对象的notify() 方法 或者 notifyAll()
方法,或者超过指定的时间量前,导致当前线程等待。

wait()后,线程会释放掉它所占有的“锁标志”,从而使线程所在对象中的其他shnchronized数据可被别的线程使用。

wait()h和notify()因为会对对象的“锁标志”进行操作,所以他们必需在Synchronized函数或者 synchronized block 中进行调用。如果在non-synchronized 函数或 non-synchronized
block 中进行调用,虽然能编译通过,但在运行时会发生IllegalMonitorStateException的异常。

关键字synchronized

该关键字用于保护共享数据,当然前提条件是要分清哪些数据是共享数据。每个对象都有一个锁标志,当一个线程访问到该对象,被Synchronized修饰的数据将被"上锁",阻止其他线程访问
。当前线程访问完这部分数据后释放锁标志,其他线程就可以访问了。

 有一个易错的地方,当调用t.sleep()的时候,会暂停线程t。这是不对的,因为Thread.sleep是一个静态方法,它会使当前线程而不是线程t进入休眠状态。例如有两个线程同时执行(没有synchronized)一个线程优先级为MAX_PRIORITY,另一个为MIN_PRIORITY,如果没有Sleep()方法,只有高优先级的线程执行完毕后,低优先级的线程才能够执行 ;但是高优先级的线程sleep(500)后,低优先级就有机会执行了。

线程优先级低并不意味着得不到处理器执行,而只是执行的频率低一点而已。而且线程的优先级一般不用我们程序员去主动设置,所以这两个方法对我们来说基本没啥用哈哈~。
在一个运行系统中,如果较高优先级的线程没有调用sleep方法,也没有受到I/O阻塞,那么较低优先级线程只能等待所有较高优先级的线程运行结束,方可有机会运行。
wait()和notify()、notifyAll() 这三个方法都是java.lang.Object的方法。
这三个方法用于协调多个线程对共享数据的存取,所以必须在synchronized语句块内使用。synchronized关键字用于保护共享数据,阻止其他线程对共享数据的存取,但是这样程序的流程就
很不灵活了,如何才能在当前线程还没退出synchronized数据块时让其他线程也有机会访问共享数据呢?此时就用这三个方法来灵活控制。

wait()方法使当前线程暂停执行并释放对象锁标示
,让其他线程可以进入synchronized数据块,当前线程被放入对象等待池中。当调用notify()方法后,将从对象的等待池中移走一个任意的线程并放到锁标志等待池中,只有锁标志等待池中
线程能够获取锁标志;如果锁标志等待池中没有线程,则notify()不起作用。

notifyAll()则从对象等待池中移走所有等待那个对象的线程并放到锁标志等待池中。

猜你喜欢

转载自www.cnblogs.com/hongchengshise/p/10404355.html