简介
写程序的过程中,会遇到很多问题。比如:你写了个接口,然后用户输入的接口参数并不满足你的要求,导致程序出了一些问题,这个时候我就需要想办法处理这些问题。
软件程序在运行过程中,非常可能遇到异常问题,这种异常问题在程序中叫异常,英文是Exception,意思是例外。程序中要做的就是合理的设计解决异常,让程序不崩溃
异常的分类
异常主要分为以下三种类型:
检查性异常:最具代表的检查性异常是用户错误或问题引起的异常,这是程序员无法预见的。例如要打开一个不存在的文件时,一个异常就发生了,这些异常在编译时不能被简单的忽略。
运行时异常:运行时异常是可能被程序员避免的异常。与检查型异常相反,运行时异常可以在编译时被忽略。
错误:错误不是一场,而是脱离程序员控制的问题。
异常体系结构
Java把异常当作对象来处理,并定义一个基类java.lang.Throwable作为所有异常的超类。
Java API中已经定义了许多异常类,这些异常类分为两大类,错误Error和异常Exception
Error 大多数情况是虚拟机抛出的,大多数情况与编写者无关也有关,需要视情况调优
VirtulMachineError 虚拟机异常
StackOverflowError 函数调用栈太深了,检查代码是否有循环调用方法而无法退出的情况,例如:递归没有结束判断就会发生该问题,原因:栈是有最大栈深度的,当你无限的入栈,栈空间肯定会溢出报错
OutOfMemoryError 在虚拟机扩展栈是无法申请到足够的空间,则抛出该异常,例如:多线程环境下,所有线程都在执行任务并且都没有及时回收,从而不断申请内存,导致扩展栈时无法申请到足够的空间
AWTError GUI图形界面化编程相关异常
Exception
IOException IO流相关异常
EOFException(END OF FILE)本质就是个消息,告诉你文件到了结尾的地方
FileNotFoundException 访问了不存在的文件或者访问到了一个目录而不是文件,就会抛出这个异常
RuntimeException 运行时异常
ArrithmeticException 算数异常,比如除数为零的时候
MissingResourceException 资源缺失异常,没加载依赖包就先执行程序导致的
ClassNotFoundException 类没找到,基本就是名字写错了,类放错了位置
NullPointerException 空指针异常
IllegalArgumentException 不合法的参数异常
ArrayIndexOutOfBoundsException 数组下标越界异常
UnkownTypeException 遇到未知种类的类型
区别
Error通常是灾难性的错误,因为发生该错误的时候,程序是无法控制的,JVM会选择终止线程;Exception通常是可以被程序处理的,并且在程序中应该尽可能的去处理这些异常。
异常处理
五个关键字:try、catch、finally、throw、throws
案例:
public class ErrorController {
public static void main(String[] args) {
try{
new ErrorController().deom(0,1);
}catch(ArithmeticException e){
e.printStackTrace();
}
}
// throws 如果这个方法处理不了这个异常,可以吐出异常,由上层调用者解决异常
public void demo(int a,int b) throws ArithmeticException{
if(a == 0){
// throw 主动抛出异常
throw new ArithmeticException();
}
//监控代码
try{
System.out.println(b/a);
//捕获异常 ArithmeticException--计算异常
}catch (ArithmeticException e){
System.out.println("计算出现问题");
// catch可以写多个,捕获多个异常,但必须从小到大捕获
}catch (Throwable t){
System.out.println("我能捕获所有异常");
// 不管有没有异常都执行该代码,处理善后
}finally {
System.out.println("程序执行完毕");
}
}
}
自定义异常
使用Java内置的异常类可以描述在编程时出现的大部分情况,当然也可以自定义异常,只需要继承Exception类即可
在程序中使用自定义异常类,大体可以分为一下几个步骤:
创建自定义异常类
在方法中通过throw关键字抛出异常对象
如果在当前抛出的异常的方法中处理异常,可以使用try-catch语句捕获并处理;否则在方法的声明处通过throws关键字指明要抛出给方法调用者的异常,继续进行下一步操作
在出现异常方法的调用者中捕获并处理异常
// 自定义异常类
public class MessageException extends Exception {
private String message;
public MessageException(String message) {
this.message = message;
}
@Override
public String toString() {
return "MessageException{"+"message='" + message + ''' +'}';
}
}
// 调用自定义异常
public static void main(String[] args){
int a = 0;
int b = 1;
//监控代码
try{
System.out.println(b/a);
//捕获异常
}catch (ArithmeticException e){
try {
throw new MessageException("错了啊");
} catch (MessageException messageException) {
System.out.println(messageException); // 输出 MessageException{message='错了啊'}程序执行完毕
}
}finally {
System.out.println("程序执行完毕");
}
}
常见问题及建议
所有的异常中最常见的应该就是空指针异常NullPointerExpetion,对于空指针异常应该遵循早暴露,早修复的原则,禁止使用catch隐藏该异常
应该养成良好的习惯来降低NullPointerException的产生,例如:
// 成员变量初始化时应该使用""而不是null
private String name = "";
// 返回数组的时候,应该返回空数组而不是null
return new String[0]
// 如果非要用null作为返回值,应该是用Optional.empty();
return Option.empty();// 这样调用方需要使用Optional.isPresent()判断是否有结果
总结
处理运行时异常时,采用逻辑去合理规避同时辅助try-catch处理
在多重catch块后面,可以加一个catch(Exception)来处理可能会被遗漏的其他异常
对于不确定的代码,也可以加上try-catch处理潜在的异常
尽量去处理异常,切记只是简单的调用printStackTrace()去打印
具体如何处理异常,要根据不同的业务需求和异常类型去决定
尽量添加finally语句块去释放占用的资源
摘录地址:
https://zhuanlan.zhihu.com/p/252763364
https://xie.infoq.cn/article/3b36d7b4c0aa0c53bff2ae9b5