对于线程的理解可以想象成每个人,就跟人一样每个人都是有状态的,比如这个人刚出生,刚睡醒,开始跑了,被绑住了,在等待,死亡都可以用来理解线程的状态。
目录
一、线程的六种状态
我们查看线程源码会发现线程的六大状态
NEW 新建状态,就是线程在start方法没有执行前就属于NEW状态
RUNNABLE 可以运行状态,线程启动了,等待CPU时间片的切换让其运行,等待资源调度
BLOCKED 阻塞状态,使用synchronized或者lock时没有获得锁时该线程就是锁状态
WAITING 等待状态,使用Thread.join()或者.wait()时,就进入了等待状态
TIMED_WAITING 计时等待,也就是说设定了等待时间,sleep(时间),join(时间),wait(时间)
TERMINATED 终止状态,线程中断或者停止运行的状态
public enum State { /** * Thread state for a thread which has not yet started. */ NEW, RUNNABLE, BLOCKED, WAITING, TIMED_WAITING, TERMINATED; }
二、用代码测试这些状态是在什么情况下发生的
我们使用代码对线程的状态进行一步步的监控,来更加深入的理解线程状态的触发方式,用休眠主线程的方式不断地判断子线程的状态方式,对于wait()我们可以用一个新线程进行唤醒,这样就可以顺利进入到终止状态,关于锁状态,这里就没演示了,可以用多个线程调用同一个方法实现这个锁状态。
/** * @author xiaomianyang * @description * @date 2020-02-07 12:30 */ public class ThreadState { public static void main(String[] args) throws InterruptedException { ThreadState threadState=new ThreadState(); Thread thread=new Thread(()->{ synchronized (threadState){ try { for(int i=0;i<10;i++) { if(i==0){ Thread.sleep(1000); }else if(i==1){ threadState.wait(); }else if(i==2){ System.out.println("我复活了"); } } } catch (InterruptedException e) { e.printStackTrace(); } } }); Thread thread2=new Thread(()->{ synchronized (threadState){ threadState.notify(); } }); System.out.println("新建状态:"+thread.getState()); thread.start(); System.out.println("等待运行状态:"+thread.getState()); Thread.sleep(500); System.out.println("计时等待状态:"+thread.getState()); Thread.sleep(1500); System.out.println("等待状态:"+thread.getState()); thread2.start(); Thread.sleep(1000); System.out.println("终止状态:"+thread.getState()); } }
输出结果如下,可以清楚看到每种状态的输出结果
新建状态:NEW 等待运行状态:RUNNABLE 计时等待状态:TIMED_WAITING 等待状态:WAITING 我复活了 终止状态:TERMINATED
三、sleep和wait的区别
- sleep和wait的显著区别就是sleep不会释放锁而wait会释放锁。
- sleep可以在任何地方使用,wait,notify,notifyAll只能在同步控制方法或者同步控制块使用
wati()和notify()可以用来干嘛呢?简单解释就是每个对象都有wait方法,比如有一个用户对象,然后有一个线程专门将用户对象里面的内容写入到数据库,这个用户对象是由另一个线程不断搜索查询过来的,所以是不固定的,有时候user对象是有值的有时候没有,如果我这个写数据库的线程每次都因为user对象没有值还在不断地运行那岂不是浪费资源码?这时候就可以判断后使用wait让这个线程处于等待状态,另一个线程将user信息写入完成后在唤醒这个写数据库的线程,当然这种应用场景会很多,这里只是简单举例。
那sleep可以让一个线程休眠从而让出CPU资源取调度其他线程,不会释放锁,也就是其他线程不能调用被sleep的方法块,除非sleep时间失效。
对于面试多线程也经常会问到sleep和wait的区别,如果不仔细研究的确会一脸懵逼,但确实有显著差别
四、wait和notify以及notifyAll的联系
wait是让一个对象等待,notify就是将等待该对象的线程唤醒,notifyAll则是唤醒所有等待该对象的线程