
- 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之间产生竞态条件。
如何避免死锁?
死锁是指两个或两个以上的进程在执行过程中,因争夺资源而造成的一种互相等待的现象,若无外力作用,它们都将无法推进下去。这是一个严重的问题,因为死锁会让你的程序挂起无法完成任务,死锁的发生必须满足以下四个条件:
互斥条件:一个资源每次只能被一个进程使用。
- 请求与保持条件:一个进程因请求资源而阻塞时,对已获得的资源保持不放。
- 不剥夺条件:进程已获得的资源,在末使用完之前,不能强行剥夺。
- 循环等待条件:若干进程之间形成一种头尾相接的循环等待资源关系。
避免死锁最简单的方法就是阻止循环等待条件,将系统中所有的资源设置标志位、排序,规定所有的进程申请资源必须以一定的顺序(升序或降序)做操作来避免死锁。
