Exception
和 Error
都是继承自 Throwable
,在Java中只有 Throwable
的实例才可以被抛( throw
)出或捕获( catch
),它是java异常处理机制的基本组成类型。
Exception
和 Error
体现了Java平台设计者对不同异常情况的分类
Exception
是程序正常运行中,可以预料的意外情况,可能并且应该被捕获从而进行相应的处理 Exception
又分为编译器异常和运行期异常。
NullPointerException
, ArrayIndexOutOfBoundsException
之类,这类异常通常都是可以通过编码避免的逻辑错误,具体根据需要来判断是非需要补货,并不会在编译期强制要求 Error
是指在正常情况下,不大可能出现的情况,绝大部分的Error都会导致程序(比如JVM自身)处于正常、不可恢复的状态。所以既然是非正常情况,那么也不便于也不需要捕获,常见的比如 OutOfMemoryError
之类,都是Error的子类。
在程序运行过程中通过某个类的字符串名称加载该类时,如果没有找到具有指定名称的类的定义的时候就会抛出 ClassNotFoundException
异常。通过字符串限定类名加载类主要有如下三种方式:
Class.forName(java.lang.String) ClassLoader.findSystemClass(java.lang.String) ClassLoader.loadClass(java.lang.String, boolean)
如果 JVM
或者 ClassLoader
实例尝试加载(可以通过正常的方法调用,也可能是使用new来创建新的对象)类的时候却找不到类的定义。要查找的类在编译的时候是存在的,运行的时候却找不到了。这个时候就会导致 NoClassDefFoundError
造成该问题的原因可能是打包过程漏掉了部分类,或者jar包出现损坏或者篡改。解决这个问题的办法是查找那些在开发期间存在于类路径下但在运行期间却不在类路径下的类。
RuntimeException
扩散出来,而不是被捕获 //下面就是一个吞掉异常的例子 try { //业务逻辑。。。 } catch(IOException e) { e.printStackTrace(); }
在处理java异常的方法中 try-catch-finally
是最常用的方式,但是在使用该代码块的时候会有一些隐藏的问题,主要是在方法返回值的问题上。
按照一般的惯性认知:当遇到return语句的时候,执行函数会立刻返回。但是在java中遇到finally的时候就有会例外。
除了return语句,try代码块中的break和continue语句也可能使控制权进入finally代码块。
当函数返回值是引用类型同时存在finally语句块时要特别注意,如果在finally块中对返回的引用对象属性进行了更改,这时即使没有在finally块中显示调用return语句,这个更改也会作用在返回值上。
public User getUser() { User user = null; try { user = new User("user"); //其他业务逻辑。。。 return user; } catch(IOException e) { //... } finally { if(user != null) { user.setName("user1"); } } }
上述代码中该函数最终返回的User对象的name属性其实为user1
产生上述问题的原因是因为JVM会保证在return语句执行之前,先执行finally语句
另外一个使用 try-catch
要注意的点是 try-catch
代码段会产生额外的性能开销,它往往会影响JVM对代码进行优化,所以建议仅捕获有必要的代码段,不要一个 try-catch
包住大段的代码。因此使用异常控制代码流程也不是一个好注意,它远比通常意义上的条件语句( if/else
, swich
)要低效
原创文章,严禁随意转载。欢迎大家添加个人微信讨论交流,添加时请备注:博客。