image.png

  • New: 线程被创建, 但没有被启动
  • Runnable: 调用start()在JVM中收集内存, 调用run()方法执行具体的逻辑, 当Thread的run方法补执行时, 线程的状态会变成Runnable, Runnable状态的线程才有资格被JVM执行, JVM并不区分有资格执行的线程和正在被执行的线程, 这两种线程都是Runnable状态
  • Blocked: 在这个状态时表示这个线程正在等待获取锁
  • Waiting: 在这个状态的线程表示在等待其它线程的动作. 比如线程执行join()方法后, 执行这个方法的线程会进入Waiting状态, 直到join()所属线程执行完成
  • Timed waiting: 这个状态和waiting状态很相似, 除了它设置了最大的等待时间
  • Terminated: 当线程执行完run()方法后会进入这个状态

    sleep()和wait()有什么区别?

  • sleep是线程类Thread的方法,导致此线程暂停执行指定时间,给执行机会给其他线程,但是监控状态依然保持,到时后会自动恢复。调用sleep不会释放对象锁。

  • wait是Object类的方法,对此对象调用wait方法导致本线程放弃对象锁,进入等待此对象的等待锁定池,只有针对此对象发出notify方法(或notifyAll)后本线程才进入对象锁定池准备获得对象锁进入运行状态。

    为什么wait和notify方法要在同步块中调用?

    主要是因为Java API强制要求这样做,如果你不这么做,你的代码会抛出IllegalMonitorStateException异常。还有一个原因是为了避免wait和notify之间产生竞态条件。

    如何避免死锁?

    死锁是指两个或两个以上的进程在执行过程中,因争夺资源而造成的一种互相等待的现象,若无外力作用,它们都将无法推进下去。这是一个严重的问题,因为死锁会让你的程序挂起无法完成任务,死锁的发生必须满足以下四个条件:

  • 互斥条件:一个资源每次只能被一个进程使用。

  • 请求与保持条件:一个进程因请求资源而阻塞时,对已获得的资源保持不放。
  • 不剥夺条件:进程已获得的资源,在末使用完之前,不能强行剥夺。
  • 循环等待条件:若干进程之间形成一种头尾相接的循环等待资源关系。

避免死锁最简单的方法就是阻止循环等待条件,将系统中所有的资源设置标志位、排序,规定所有的进程申请资源必须以一定的顺序(升序或降序)做操作来避免死锁。