方式一
try....catch...finally,在catch中捕获、处理异常
方式二
设置默认的全局异常处理器。
发生异常时,一级一级往上抛,
这个过程中,如果异常被catch捕获、处理,那就没事了;
如果没有对应的catch来捕获、处理异常,最终被抛到DispatcherServlet、web服务器,web服务器的处理方式是:传回一堆错误信息显示在浏览器中给用户看。
我们可以设置一个默认的全局异常处理器,抛到DispatcherServlet时,由DispatcherServlet调用默认的全局异常处理器来处理异常。
相当于设置了一个全局的catch。
新建一个类,实现HandlerExceptionResolver接口即可:
@Component public class MyExceptionResolver implements HandlerExceptionResolver { @Override public ModelAndView resolveException(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Object o, Exception e) { ModelAndView mav = new ModelAndView("error"); mav.addObject("msg", e.getMessage()); return mav; } }
- 需要把这个类放到Spring容器中,可以用<bean>来配置,也可以用@Component标注。
- 会自动传入异常对象,我们可以从这个异常对象中获取异常信息
- 返回值是ModelAndView类型,这个类会在jvm之前拦截没人要的异常,调用resolveException()来处理,并用指定的视图来响应。可以向视图传递异常信息。
原本会在浏览器页面上显示一堆错误信息,全是专业名词、还是英文,对用户很不友好。
此种方式,可以指定视图、自定义要显示的信息,对用户很友好。
缺点是统一处理未经捕获的异常,不管什么样的异常,处理方式都一样,太笼统。
方式三
使用自定义的异常类。
(1)新建包com.chy.exception,用来放自定义的异常类。然后编写多个自定义的异常类。
比如转账,可能出现:
- 余额不足,NoMoneyException
- 对方账户不存在,UserNotFoundException
- 账户被冻结,FrozenException
public class NoMoneyException extends Exception{ private String msg; public NoMoneyException(String msg) { super(); this.msg = msg; } public String getMsg() { return msg; } }
提供String类型的字符串来表示异常信息,对应的构造函数来创建异常对象,提供对应的getter方法来获取异常信息。
以同样的方式创建其它2个异常。
alt+insert时,不要使用String message,这个是基类Exception的参数,表示代码执行的错误信息。用我们自己写的变量就OK。
我们自定义的异常不是为了处理代码错误,代码错误直接try...catch就好了,没必要自定义异常。
自定义异常用于友好提示用户,比如说dao层查询到余额不足,我直接抛一个NoMoneyException,代码执行本身是没有错误的,异常是手动抛的。
(2)在dao层抛出异常
public class UserDao{ public void transfer(int to,int from,double amount) throws NoMoneyException,UserNotFoundException,FrozenException { //此处省略连接数据库查询 //..... //余额不足 if (余额不足){ throw new NoMoneyException("您当前余额xx元,余额不足,无法完成转账!"); } //对方账号不存在 if (对方账号不存在){ throw new UserNotFoundException("对方账号xxxxxxxxxx不存在!"); } //对方账号被冻结 if (对方账号被冻结){ throw new FrozenException("对方账号xxxxxxxxxx已被冻结,无法向对方转账!"); } //.... } }
不使用catch来处理,要在方法签名上抛出异常。
在对应的service层的方法签名上也要抛出来:
public class UserService{ public void transfer(int to,int from,double amount) throws NoMoneyException,UserNotFoundException,FrozenException { //调用dao层 } }
(3)在controller中处理异常
@org.springframework.stereotype.Controller public class UserController { @RequestMapping("/transfer") public ModelAndView transfer() { try { //调用service层 }catch (NoMoneyException e){ //可以调用e.getMsg()获取自定义的异常信息,转发到某个视图、传递数据 //..... }catch (UserNotFoundException e){ //.... }catch (FrozenException e){ //...... } } }
也可以在全局异常类中处理:
@Component public class MyExceptionResolver implements HandlerExceptionResolver { @Override public ModelAndView resolveException(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Object o, Exception e) { //按异常的种类来进行对应的处理, if (e instanceof NoMoneyException){ //..... } if (e instanceof UserNotFoundException){ //..... } if (e instanceof FrozenException){ //..... } } }
如果处理方式都相同,比如都是转发到某个视图、显式自定义的错误信息,那就不必写if判断来区分异常类型,直接写处理。
同样可以调用视图、传递数据。
方式三通常和try...catch配合使用:
- try...catch处理代码错误,比如文件找不到、空指针异常等,代码执行本身出了问题;
- 方式三主要用于我们自定义的错误,代码执行本身没问题。