文章插图
线程在生命周期中并不是固定处于某一个状态而是随着代码的执行在不同状态之间切换 。Java 线程状态变迁如下图所示(图源《Java 并发编程艺术》4.1.4节):

文章插图
由上图可以看出:
线程创建之后它将处于 NEW(新建) 状态,调用 start() 方法后开始运行,线程这时候处于 READY(可运行) 状态 。可运行状态的线程获得了 cpu 时间片(timeslice)后就处于 RUNNING(运行) 状态 。
操作系统隐藏 Java虚拟机(JVM)中的 RUNNABLE 和 RUNNING 状态,它只能看到 RUNNABLE 状态(图源:HowToDoInJava:Java Thread Life Cycle and Thread States),所以 Java 系统一般将这两个状态统称为 RUNNABLE(运行中) 状态。

文章插图
当线程执行 wait()方法之后,线程进入 WAITING(等待)状态 。进入等待状态的线程需要依靠其他线程的通知才能够返回到运行状态,而 TIME_WAITING(超时等待) 状态相当于在等待状态的基础上增加了超时限制,比如通过 sleep(long millis)方法或 wait(long millis)方法可以将 Java 线程置于 TIMED WAITING 状态 。当超时时间到达后 Java 线程将会返回到 RUNNABLE 状态 。当线程调用同步方法时,在没有获取到锁的情况下,线程将会进入到 BLOCKED(阻塞) 状态 。线程在执行 Runnable 的 run() 方法之后将会进入到 TERMINATED(终止) 状态 。
31 关于 final 关键字的一些总结final关键字主要用在三个地方:变量、方法、类 。
- 对于一个final变量,如果是基本数据类型的变量,则其数值一旦在初始化之后便不能更改;如果是引用类型的变量,则在对其初始化之后便不能再让其指向另一个对象 。
- 当用final修饰一个类时,表明这个类不能被继承 。final类中的所有成员方法都会被隐式地指定为final方法 。
- 使用final方法的原因有两个 。第一个原因是把方法锁定,以防任何继承类修改它的含义;第二个原因是效率 。在早期的Java实现版本中,会将final方法转为内嵌调用 。但是如果方法过于庞大,可能看不到内嵌调用带来的任何性能提升(现在的Java版本已经不需要使用final方法进行这些优化了) 。类中所有的private方法都隐式地指定为final 。

文章插图
在 Java 中,所有的异常都有一个共同的祖先java.lang包中的 Throwable类 。Throwable: 有两个重要的子类:Exception(异常) 和 Error(错误) ,二者都是 Java 异常处理的重要子类,各自都包含大量子类 。
Error(错误):是程序无法处理的错误,表示运行应用程序中较严重问题 。大多数错误与代码编写者执行的操作无关,而表示代码运行时 JVM(Java 虚拟机)出现的问题 。例如,Java虚拟机运行错误(Virtual MachineError),当 JVM 不再有继续执行操作所需的内存资源时,将出现 OutOfMemoryError 。这些异常发生时,Java虚拟机(JVM)一般会选择线程终止 。
这些错误表示故障发生于虚拟机自身、或者发生在虚拟机试图执行应用时,如Java虚拟机运行错误(Virtual MachineError)、类定义错误(NoClassDefFoundError)等 。这些错误是不可查的,因为它们在应用程序的控制和处理能力之 外,而且绝大多数是程序运行时不允许出现的状况 。对于设计合理的应用程序来说,即使确实发生了错误,本质上也不应该试图去处理它所引起的异常状况 。在 Java中,错误通过Error的子类描述 。
Exception(异常):是程序本身可以处理的异常 。</font>Exception 类有一个重要的子类 RuntimeException 。RuntimeException 异常由Java虚拟机抛出 。NullPointerException(要访问的变量没有引用任何对象时,抛出该异常)、ArithmeticException(算术运算异常,一个整数除以0时,抛出该异常)和 ArrayIndexOutOfBoundsException (下标越界异常) 。
注意:异常和错误的区别:异常能被程序本身可以处理,错误是无法处理 。
Throwable类常用方法
- public string getMessage():返回异常发生时的详细信息
- public string toString():返回异常发生时的简要描述
- public string getLocalizedMessage():返回异常对象的本地化信息 。使用Throwable的子类覆盖这个方法,可以声称本地化信息 。如果子类没有覆盖该方法,则该方法返回的信息与getMessage()返回的结果相同
推荐阅读
- 用11个事实为8岁女儿解读深度学习
- 阿里大牛为学习java的菜鸟准备的攻略!
- 我的文档在哪找 我的文档在哪
- 收藏铁壶的六条必知常识
- Java仿QQ生成图片验证码
- javaAPI解读
- 疯狂英语教你快速学英语的秘方 李阳疯狂英语学习方法
- Java中使用Optional有什么好处
- Java配置文件敏感信息处理
- Java变量作用域的这4条铁规
