SpringMVC的异常处理

 

方式一

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处理代码错误,比如文件找不到、空指针异常等,代码执行本身出了问题;
  • 方式三主要用于我们自定义的错误,代码执行本身没问题。

 

posted @ 2020-02-02 21:29  chy_18883701161  阅读(467)  评论(0编辑  收藏  举报