try....catch...finally,在catch中捕获、处理异常web
设置默认的全局异常处理器。spring
发生异常时,一级一级往上抛,数据库
这个过程当中,若是异常被catch捕获、处理,那就没事了;浏览器
若是没有对应的catch来捕获、处理异常,最终被抛到DispatcherServlet、web服务器,web服务器的处理方式是:传回一堆错误信息显示在浏览器中给用户看。服务器
咱们能够设置一个默认的全局异常处理器,抛到DispatcherServlet时,由DispatcherServlet调用默认的全局异常处理器来处理异常。app
至关于设置了一个全局的catch。jvm
新建一个类,实现HandlerExceptionResolver接口便可:ide
@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; } }
本来会在浏览器页面上显示一堆错误信息,全是专业名词、仍是英文,对用户很不友好。函数
此种方式,能够指定视图、自定义要显示的信息,对用户很友好。this
缺点是统一处理未经捕获的异常,无论什么样的异常,处理方式都同样,太笼统。
使用自定义的异常类。
好比转帐,可能出现:
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,代码执行自己是没有错误的,异常是手动抛的。
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层 } }
@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配合使用: