Java异常
Throwable
在 Java 中,所有的异常都有一个共同的祖先 Throwable(可抛出)。Throwable 指定代码中可用异常传播机制通过 Java 应用程序传输的任何问题的共性。
而Throwable有两个重要的子类:Exception(异常)和 Error(错误),二者都是 Java 异常处理的重要子类,各自都包含大量子类。
Error(错误)
是程序无法处理的错误,表示运行应用程序中较严重问题。
大多数错误与代码编写者执行的操作无关,而表示代码运行时 JVM(Java 虚拟机)出现的问题。例如,Java虚拟机运行错误(Virtual MachineError),当 JVM 不再有继续执行操作所需的内存资源时,将出现内存溢出 错误OutOfMemoryError。这些异常发生时,Java虚拟机(JVM)一般会选择线程终止。 这些错误表示故障发生于虚拟机自身、或者发生在虚拟机试图执行应用时,如Java虚拟机运行错误(Virtual MachineError)、类定义错误(NoClassDefFoundError)等。这些错误是不可查的,因为它们在应用程序的控制和处理能力之 外,而且绝大多数是程序运行时不允许出现的状况。对于设计合理的应用程序来说,即使确实发生了错误,本质上也不应该试图去处理它所引起的异常状况。在 Java中,错误通过Error的子类描述。
Exception (异常)
&esmp; 是程序本身可以处理的异常。
&esmp; Exception 类有一个重要的子类 RuntimeException。RuntimeException 类及其子类表示“JVM 常用操作”引发的错误。例如,几个常见的RuntimeException:
. | . |
---|---|
NullPointerException | 空指针引用异常 |
ClassNotFoundException | 没有找到指定类异常 |
MissingResourceException | 没有找到指定资源异常 |
ArithmeticException | 算数运算异常(分母为0) |
SecurityException | 安全异常 |
TypeNotPresentException | 类型不存在异常 |
UnsupportedOperationException | 不支持的操作异常 |
IndexOutOfBoundsException | 下标越界异常 |
NegativeArraySizeException | 创建一个大小为负数的数组异常 |
IllegalArgumentException | 传递非法参数异常 |
PS: 异常和错误的区别:异常能被程序本身可以处理,错误是无法处理。
可查的异常(checked exceptions)和不可查的异常(unchecked exceptions)
Throwable大类下面,通常还会根据“异常是否可查”分为两类,即可查的异常(checked exceptions) 和 不可查的异常(unchecked exceptions)。
可查的异常(checked exceptions):指的是编译器要求必须处理的异常;可查异常虽然是异常状况,但在一定程度上它的发生是可以预计的,而且一旦发生这种异常状况,就必须采取某种方式进行处理。 除了RuntimeException及其子类以外,其他的Exception类及其子类都属于可查异常。这种异常的特点是Java编译器会检查它,也就是说,当程序中可能出现这类异常,要么用try-catch语句捕获它,要么用throws子句声明抛出它,否则编译不会通过。
不可查的异常(unchecked exceptions): 指的是编译器不要求处理的异常;包括运行时异常(RuntimeException与其子类)和错误(Error)。
运行时异常 、 非运行时异常
Exception大类下面,通常还会根据“是否在运行时抛出”分为两类,即运行时异常 和 非运行时异常。
运行时异常:都是RuntimeException类及其子类异常,如NullPointerException(空指针异常)、IndexOutOfBoundsException(下标越界异常)等,这些异常是不可查异常,程序中可以选择捕获处理,也可以不处理。这些异常一般是由程序逻辑错误引起的,程序应该从逻辑角度尽可能避免这类异常的发生。运行时异常的特点是Java编译器不会检查它,也就是说,当程序中可能出现这类异常,即使没有用try-catch语句捕获它,也没有用throws子句声明抛出它,也会编译通过。
非运行时异常 (编译异常):是RuntimeException以外的异常,类型上都属于Exception类及其子类。从程序语法角度讲是必须进行处理的异常,如果不处理,程序就不能编译通过。如IOException、SQLException等以及用户自定义的Exception异常,一般情况下不自定义检查异常。
异常处理机制
在 Java 应用程序中,异常处理机制为:抛出异常,捕捉异常。
抛出异常:当一个方法出现错误引发异常时,方法创建异常对象并交付运行时系统,异常对象中包含了异常类型和异常出现时的程序状态等异常信息。运行时系统负责寻找处置异常的代码并执行。
捕获异常:在方法抛出异常之后,运行时系统将转为寻找合适的异常处理器(exception handler)。潜在的异常处理器是异常发生时依次存留在调用栈中的方法的集合。当异常处理器所能处理的异常类型与方法抛出的异常类型相符时,即为合适 的异常处理器。运行时系统从发生异常的方法开始,依次回查调用栈中的方法,直至找到含有合适异常处理器的方法并执行。当运行时系统遍历调用栈而未找到合适 的异常处理器,则运行时系统终止。同时,意味着Java程序的终止。
自定义异常
方式一:
在Java程序中,我们可以通过继承父类RuntimeException, 然后定义一个构建方法,在构建方法里面调用父类的构建方法super(message)即可。
方式二:
在使用Spring MVC的情况下, 可以直接在自定义异常处理类中添加上@ControllerAdvice 注解;这个注解的意思是把@ControllerAdvice注解内部使用@ExceptionHandler注解的方法应用到所有的 @RequestMapping注解的方法。非常简单,不过只有当使用@ExceptionHandler最有用,另外两个用处不大。所以也就是需要在实现类的方法上,添加上@ExceptionHandler(Exception.class)的注解。
加上@ResponseBody可直接将返回对象转换成JSon格式。
具体实现如下:
@ExceptionHandler(Exception.class)
@ResponseBody
public ResultJson resolveException(HttpServletRequest request,
HttpServletResponse response, Object handler, Exception ex) {
Map<String, Object> model = new HashMap<String, Object>();
model.put("ex", ex);
response.setContentType("application/json;charset=UTF-8");
ResultJson resultJson = new ResultJson();
if (ex instanceof ApplicationException) {
resultJson.setState(false);
resultJson.setMessage(ex.getMessage());
resultJson.setFlag("");
resultJson.setResult("");
} else if (ex instanceof SecurityException) {
int errCode = ex.getCause() == null ? 201 : Integer.valueOf(ex.getCause().getMessage());
resultJson.setErrorCode(errCode);
resultJson.setState(false);
resultJson.setMessage(ex.getMessage());
resultJson.setFlag("");
resultJson.setResult("");
logger.info("--------发生错误--------");
logger.info("错误信息:"+ex.getMessage());
logger.info("错误编码:"+errCode);
} else {
resultJson.setErrorCode(201);
resultJson.setState(false);
resultJson.setMessage("系统异常,操作失败!");
resultJson.setFlag("");
resultJson.setResult("");
}
ex.printStackTrace();
logger.info("--------发生错误--------");
StringBuilder str = new StringBuilder("参数: ");
Set<String> keySet = request.getParameterMap().keySet();
if(!keySet.isEmpty()) {
for(String key : keySet) {
str.append(key).append(":").append(request.getParameter(key)).append(" ");
}
}
return resultJson;
}