SpringMvc异常处理机制

107 阅读3分钟

1、Java的异常处理思路及其在业务层的弊端

一般情况下,我们处理异常都是使用try-catch进行捕捉,但在业务层(controller层)进行try-catch有一定的弊端:

  1. try-catch这个动作和业务代码耦合到一起了
  2. 且如果在方法1中需要进行try-catch对”类型转换异常“进行捕捉,其他方法也有可能需要捕捉”类型转换异常“,会导致代码冗余--抽取思想来解决,比如,就用某一个组件来单独解决”类型转换异常“

2、异常处理思路--SpringMvc

系统中异常包括两类:预期异常运行时异常RuntimeException,前者通过捕获异常从而获取异常信息,后者主要通过规范代码开发、测试等手段减少运行时异常的发生。 系统的Dao、Service、Controller出现都通过throws Exception向上抛出,最后由SpringMVC前端控制器交由异常处理器--HandlerExceptionResolver进行异常处理,如下图:

image.png

3、异常处理的两种方式

3.1 SpringMVC提供的简单异常处理器SimpleMappingExceptionResolver

SpringMVC已经定义好了该类型转换器,在使用时可以根据项目情况进行相应异常与视图的映射配置

在spring-mvc.xml中的配置:

<!-- 在spring-mvc.xml中的配置 -->
<bean class="org.springframework.web.servlet.handler.SimpleMappingExceptionResolver">
        <!--默认的错误视图,已经配置过内部资源视图解析器,因此只需要返回视图名称-->
        <property name="defaultErrorView" value="error"/>
        <!--配置异常类型(可配置多个)-->
        <property name="exceptionMappings">
            <map>
                <!--key="异常类型" value="错误视图"-->
                <!--发生MyException异常就跳往error1界面
                    发生ClassCastException异常就跳往error2界面-->
                <entry key="com.kkk.exception.MyException" value="error1"/>
                <entry key="java.lang.ClassCastException" value="error2"/>
            </map>
        </property>
    </bean>

<!--配置内部资源视图解析器-->
<bean id="ResourceViewResolver" class="org.springframework.web.servlet.view.InternalResourceViewResolver">
    <!--配置内部资源视图解析器,因此在默认的错误视图中直接填写value="error/"即可-->
    <property name="prefix" value="/"/>
    <property name="suffix" value=".jsp"/>
</bean>

3.2 实现Spring的异常处理接口HandlerExceptionResolver,自定义自己的异常处理器

实现步骤

①创建异常处理器类实现HandlerExceptionResolver

public class MyResolver implements HandlerExceptionResolver {
    @Override
    /**
     * 参数Exception e:异常对象
     * 返回值ModelAndView:跳转到错误视图信息
     */
    public ModelAndView resolveException(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Object o, Exception e) {
        ModelAndView modelAndView = new ModelAndView();
       //MyException异常,执行modelAndView.addObject("info","自定义异常");操作
        if(e instanceof MyException){
            modelAndView.addObject("info","自定义异常");
        }else if(e instanceof ClassCastException){
            modelAndView.addObject("info","类转换异常");
        }
        modelAndView.setViewName("error");
        return modelAndView;
    }
}

②spring-mvc.xml配置异常处理器

    <!--配置自定义异常处理器-->
    <bean class="com.kkk.resolver.MyResolver"/>

③编写异常页面

<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>Title</title>
</head>
<body>
    <h1>通用的错误提示页面</h1>
    <h1>${info}</h1>
</body>
</html>

④测试异常跳转

@Controller
public class DemoController {
    @Autowired
    private DemoService demoService;
    /**@RequestParam 是传递参数的.
     * @RequestParam 用于将请求参数区数据映射到功能处理方法的参数上。
     *在url中输入:localhost:8080/.../'?'userName=zhangsan
     * 请求中包含username参数(如/requestparam1?userName=zhang),则自动传入。
     * 接下来我们看一下@RequestParam注解主要有哪些参数:
     * value:参数名字,即入参的请求参数名字,如username表示请求的参数区中的名字为username的参数的值将传入;
     * required:是否必须,默认是true,表示请求中一定要有相应的参数,否则将报404错误码;
     * defaultValue:默认值,表示如果请求中没有同名参数时的默认值,默认值可以是SpEL表达式,如“#{systemProperties['java.vm.version']}”。
     */
    @RequestMapping("/errorTest")
    public String show(@RequestParam(value = "name")String name){
        System.out.println("show is running...");
        demoService.show1();
//        demoService.show2();
//        demoService.show3();
//        demoService.show4();
//        demoService.show5();
        return "index";
    }
}
@Service("demoService")
public class DemoServiceImpl implements DemoService {
    @Override
    public void show1() {
        System.out.println("抛出类型转换异常....");
        Object str = "zhangsan";
        Integer num = (Integer)str;
    }
    @Override
    public void show2() {
        System.out.println("抛出除零异常....");
        int i = 1/0;
    }
    @Override
    public void show3() throws FileNotFoundException {
        System.out.println("文件找不到异常....");
        InputStream in = new FileInputStream("C:/xxx/xxx/xxx.txt");
    }
    @Override
    public void show4() {
        System.out.println("空指针异常.....");
        String str = null;
        str.length();
    }
    @Override
    public void show5() throws MyException {
        System.out.println("自定义异常....");
        throw new MyException();
    }
}