1. New(新建)
当用new操作符创建一个新线程时,如 new Thread(r), 该线程还没有开始运行。这意味着它的状态是new,未调用start方法。
2. Runnable(可运行)
一旦调用start()方法,线程就处于runnable状态。可以可运行的线程可能正在运行也可能没有运行,这取决于操作系统给线程提供运行的时间(线程获取cpu的执行权限,所以这个状态其实代表了2种状态,runnable和running状态)
事实上,运行中的线程被中断,目的是为了让他们线程获得运行机会。线程调度的细节依赖于操作系统提供的服务。抢占式调度系统给每一个可运行线程一个时间片来执行任务。当时间片用完,操作系统剥夺该线程的运行权,并给另一个线程可运行机会。当选择下一个线程时,操作系统考虑线程的优先级。
yield()的作用是让步。它能让当前线程由“运行状态”进入到“就绪状态Runnable”,从而让其它具有相同优先级的等待线程获取执行权;但是,并不能保证在当前线程调用yield()之后,其它具有相同优先级的线程就一定能获得执行权;也有可能是当前线程又进入到“运行状态”继续运行!
3. Blocked(被阻塞)
当一个线程试图获取一个内部的对象锁synchronized(而不是java.util.concurrent库里的锁), 而该锁被其他线程持有,则该线程进入阻塞状态。当其他线程释放该锁,并且线程调度器允许本线程持有它的时候,该线程将变成非阻塞状态。
4. Waiting(等待)
当线程通知另一个线程通知调度器一个条件时,它自己进入等待状态。在调用Object.wait方法或Thread.join方法,或者是等待java.util.concurrent库中的Lock或Condition时,就会出现这种情况。实际上,被阻塞状态与被等待状态是有很大不同的。
5.Timed Waiting(计时等待)
有几个方法有一个超时参数。调用它们导致线程进入计时等待(timed waiting)状态。这一状态将一直保持到超时期满或者接收到适当的通知。带有超时参数的方法有Thread.sleep和Object.wait, Thrad.join, Lock.tryLock以及Condition.await的计时版。
6. Terminated(被终止)
线程因如下两个原因之一而被终止:
1) 因为run方法正常退出而自然死亡
2) 因为一个没有捕获的异常终止了run方法二意外死亡
wait/notify/notifyall/yield必须结合synchronized使用,Thread.join底层也是wait方法
注:以上案列经过本人测试