一、JVM线程通常有六种状态
-
New(新建): 尚未启动的线程的线程状态。
-
Runnable(可运行): 可运行线程的线程状态,等待CPU调度。
-
Blocked(阻塞): 线程阻塞等待监视器锁定的线程状态。处于synchronized同步代码块或方法中被阻塞。
-
Waiting(无限期等待): 等待其它线程显式地唤醒,否则不会被分配 CPU 时间片。
-
Timed Waiting(限期等待):无需等待其它线程显式地唤醒,在一定时间之后会被系统自动唤醒。
- 调用 Thread.sleep() 方法使线程进入限期等待状态时,常常用“使一个线程睡眠”进行描述。
- 调用 Object.wait() 方法使线程进入限期等待或者无限期等待时,常常用“挂起一个线程”进行描述。
- 睡眠和挂起是用来描述行为,而阻塞和等待用来描述状态。
- 阻塞和等待的区别在于,阻塞是被动的,它是在等待获取一个排它锁。而等待是主动的,通过调用 Thread.sleep() 和 Object.wait() 等方法进入。
- Terminated(死亡): 终止线程的线程状态。线程正常完成执行或者出现异常。
二、Java中sleep,wait,yield,join的区别
1.sleep()方法
- 在指定时间内让当前正在执行的线程暂停执行,但不会释放“锁标志”.不推荐使用.
- sleep()使当前线程进入阻塞状态,在指定时间内不会执行。
2.wait()方法
- 在其他线程调用对象的notify或notifyAll方法前,导致当前线程等待。线程会释放掉它所占有的“锁标志”,从而使别的线程有机会抢占该锁。
- 当前线程必须拥有当前对象锁。如果当前线程不是此锁的拥有者,会抛出IllegalMonitorStateException异常。
- 唤醒当前对象锁的等待线程使用notify或notifyAll方法,也必须拥有相同的对象锁,否则也会抛出IllegalMonitorStateException异常。
- waite()和notify()必须在synchronized函数或synchronized block中进行调用。如果在non-synchronized函数或non-synchronized block中进行调用,虽然能编译通过,但在运行时会发生IllegalMonitorStateException的异常。
3.yield方法
- 暂停当前正在执行的线程对象。
- yield()只是使当前线程重新回到可执行状态,所以执行yield()的线程有可能在进入到可执行状态后马上又被执行。
- yield()只能使同优先级或更高优先级的线程有执行的机会。
- 调用yield方法并不会让线程进入阻塞状态,而是让线程重回就绪状态,它只需要等待重新获取CPU执行时间,这一点是和sleep方法不一样的。
4.join方法
- 等待该线程终止。
- 等待调用join方法的线程结束,再继续执行。如:t.join();//主要用于等待t线程运行结束,若无此句,main则会执行完毕,导致结果不可预测。
- 在很多情况下,主线程创建并启动了线程,如果子线程中要进行大量耗时运算,主线程往往将早于子线程结束之前结束。这时,如果主线程想等待子线程执行完成之后再结束,比如子线程处理一个数据,主线程要取得这个数据中的值,就要用到join()方法了。方法join()的作用是等待线程对象销毁。