SpringMVC-异常处理及常用组件

异常处理器

1.基于配置的异常处理

springmvc提供了一个处理控制器方法执行过程中所出现的异常的接口: HandlerExceptionResolver

HandlerExceptionResolver接口的实现类有: DefaultHandlerExceptionResolver和SimpleMappingExceptionResolver

springmvc提供了自定义的异常处理器SimpleMappingExceptionResolver, 使用方式: (注意拦截器要放行)

 
<!--配置 异常处理-->
 <bean class="org.springframework.web.servlet.handler.SimpleMappingExceptionResolver">
     <!--properties的键表示 处理器方法执行过程中出现的异常,
         properties的键表示若出现指定异常时,设置一个新的视图名称,跳转到指定页面
     -->
     <property name="exceptionMappings">
         <props>
             <prop key="java.lang.ArithmeticException">error</prop>
         </props>
     </property>
    <!-- 设置 将异常信息共享在请求域中的 键 ,也可以不设置,如果不设置 页面中 就可以不写 ${ex}-->
     <property name="exceptionAttribute" value="ex">
     </property>
 </bean>
 @RequestMapping("/testExp")
 public String testExp(){
     System.out.println(1/0);
     return "success";
 }

编写 error.html

 出现错误
 <p th:text="${ex}"></p>   显示异常错误信息

页面显示:

出现错误

java.lang.ArithmeticException: / by zero

2.基于注解的异常处理

去掉或注释掉 springmvc 配置文件的 异常的配置SimpleMappingExceptionResolver ,采用注解方式

 //@ControllerAdvice  将当前类所标识为 异常处理的组件
 @ControllerAdvice
 public class ExceptionContoller {
    //@ExceptionHandler  用于设置所标识方法处理的异常
     @ExceptionHandler(value = {ArithmeticException.class,NullPointerException.class})
     //ex 表示 当前请求处理中出现的异常对象
     public String test(Exception ex, Model model){
         model.addAttribute("ex",ex);
         return "error";
     }
 }
 @RequestMapping("/testExp")
 public String testExp(){
     System.out.println(1/0);
     return "success";
 }

页面显示:

出现错误

java.lang.ArithmeticException: / by zero

注解配置springmvc

使用配置类和注解代替 web.xml 和springmvc配置文件的功能

1.创建初始化类,代替web.xml

在Servlet3.0环境中,容器会在类路径中查找实现javaxervlet.ServletContainerInitializer接口的类,如果找到的话就用它来配置Servlet容器。

Spring提供了这个接口的实现,名为SpringServletContainerlnitializer,这个类反过来又会查找实现WebApplicationInitializer的类并将配置的任务交给它们来完成。Spring3.2引入了一个便利的 WebApplicationInitializer基础实现,名为AbstractAnnotationConfigDispatcherServletinitializer,当我们的类扩展了AbstractAnnotationConfigDispatcherServletInitializer并将其部署到Servlet3.0容嚣的时候,容器会自动发现它,并用它来配置Servlet上下文。

 /**
  *
  * web工程的初始化类,用来代替 web.xml
  */
 ​
 public class WebInit extends AbstractAnnotationConfigDispatcherServletInitializer {
     //指定spring的配置类
     @Override
     protected Class<?>[] getRootConfigClasses() {
         return new Class[]{SpringConfig.class};
     }
     //指定springmvc的配置类
     @Override
     protected Class<?>[] getServletConfigClasses() {
         return new Class[]{WebConfig.class};
     }
     // 指定dispacherservlet的映射规则
 ​
     @Override
     protected String[] getServletMappings() {
         return new String[]{"/"};
     }
     //注册 过滤器
     @Override
     protected Filter[] getServletFilters() {
         CharacterEncodingFilter characterEncodingFilter = new CharacterEncodingFilter();
         characterEncodingFilter.setEncoding("UTF-8");
         characterEncodingFilter.setForceResponseEncoding(true);
 ​
         HiddenHttpMethodFilter hiddenHttpMethodFilter = new HiddenHttpMethodFilter();
 ​
         return new Filter[]{characterEncodingFilter,hiddenHttpMethodFilter};
     }
 }
 @Configuration   //将当前类标识为一个配置类
 @ComponentScan(basePackages = "com.ly")   // 1.扫描组件
 @EnableWebMvc    // 5.开启mvc注解驱动
 public class WebConfig  {
 ​
     //3.view-controller
 ​
 ​
     //4: default-servlet-handler
 ​
     //6. 文件上传解析器
 ​
     //7. 异常处理
 ​
     //8.拦截器
 ​
     /**
      * 2.视图解析器
      * 以下 3个方式 都是 配置  视图解析器
      */
     //生成视图解析器,并 为 解析器注入模板引擎
     @Bean
     public ViewResolver viewResolver(SpringTemplateEngine templateEngine){
         ThymeleafViewResolver viewResolver = new ThymeleafViewResolver();
         viewResolver.setCharacterEncoding("UTF-8");
         viewResolver.setTemplateEngine(templateEngine);
         return viewResolver;
     }
 ​
 ​
 ​
     //生成模板引擎 并为模板引擎注入模板解析器
     @Bean
     public SpringTemplateEngine templateEngine(ITemplateResolver templateResolver){
         SpringTemplateEngine templateEngine = new SpringTemplateEngine();
         templateEngine.setTemplateResolver(templateResolver);
         return templateEngine;
     }
 ​
 ​
     //配置生成模板解析器
     @Bean
     public ITemplateResolver templateResolver(){
         WebApplicationContext webApplicationContext = ContextLoader.getCurrentWebApplicationContext();
         ServletContextTemplateResolver templateResolver = new ServletContextTemplateResolver(webApplicationContext.getServletContext());
         templateResolver.setPrefix("/WEB-INF/templates/");
         templateResolver.setSuffix(".html");
         templateResolver.setCharacterEncoding("UTF-8");
         templateResolver.setTemplateMode(TemplateMode.HTML);
         return templateResolver;
 ​
     }
 ​
 ​
 }
 //替换spring配置文件
 @Configuration
 public class SpringConfig {
 }
 @Controller
 public class TestController {
 ​
     @RequestMapping("/")
     public String index(){
         return "index";
     }
 }

在webapp/WEB-INF/templates/index.html

 <body>
 首页
 </body>

启动项目 即可访问 index.html

完整版 ---使用配置类 代替springmvc.xml

 package com.ly.config;
 ​
 import com.ly.interceptor.TestInterceptor;
 import org.springframework.context.annotation.Bean;
 import org.springframework.context.annotation.ComponentScan;
 import org.springframework.context.annotation.Configuration;
 import org.springframework.http.client.MultipartBodyBuilder;
 import org.springframework.web.context.ContextLoader;
 import org.springframework.web.context.WebApplicationContext;
 import org.springframework.web.multipart.MultipartFile;
 import org.springframework.web.multipart.MultipartResolver;
 import org.springframework.web.multipart.commons.CommonsMultipartResolver;
 import org.springframework.web.servlet.HandlerExceptionResolver;
 import org.springframework.web.servlet.ViewResolver;
 import org.springframework.web.servlet.config.annotation.*;
 import org.springframework.web.servlet.handler.SimpleMappingExceptionResolver;
 import org.thymeleaf.spring5.SpringTemplateEngine;
 import org.thymeleaf.spring5.view.ThymeleafViewResolver;
 import org.thymeleaf.templatemode.TemplateMode;
 import org.thymeleaf.templateresolver.ITemplateResolver;
 import org.thymeleaf.templateresolver.ServletContextTemplateResolver;
 ​
 import javax.swing.*;
 import java.util.List;
 import java.util.Properties;
 ​
 /**
  * 代替springmvc.xml
  * 1.扫描组件
  * 2.视图解析器
  * 3.view-controller
  * 4: default-servlet-handler
  * 5. mvc注解驱动
  * 6. 文件上传解析器
  * 7.异常处理
  * 8.拦截器
  */
 @Configuration   //将当前类标识为一个配置类
 @ComponentScan(basePackages = "com.ly")   // 1.扫描组件
 @EnableWebMvc    // 5.开启mvc注解驱动
 public class WebConfig implements WebMvcConfigurer {
 ​
     //3.view-controller
 ​
     @Override
     public void addViewControllers(ViewControllerRegistry registry) {
         registry.addViewController("/hello").setViewName("hello");
     }
 ​
     //4: default-servlet-handler
     @Override
     public void configureDefaultServletHandling(DefaultServletHandlerConfigurer configurer) {
         configurer.enable();
     }
     //8.拦截器
     @Override
     public void addInterceptors(InterceptorRegistry registry) {
         TestInterceptor tt = new TestInterceptor();
         registry.addInterceptor(tt).addPathPatterns("/**");
     }
 ​
    //6. 文件上传解析器
     @Bean
     public MultipartResolver multipartResolver(){
         CommonsMultipartResolver commonsMultipartResolver =new CommonsMultipartResolver();
         return  commonsMultipartResolver;
     }
     //7. 异常处理
     @Override
     public void configureHandlerExceptionResolvers(List<HandlerExceptionResolver> resolvers) {
         SimpleMappingExceptionResolver exceptionResolver = new SimpleMappingExceptionResolver();
         Properties prop = new Properties();
         prop.setProperty("java.lang.ArithmeticException","error");
         exceptionResolver.setExceptionMappings(prop);
         exceptionResolver.setExceptionAttribute("ex");
         resolvers.add(exceptionResolver);
     }
 ​
     /**
      * 2.视图解析器
      * 以下 3个方式 都是 配置  视图解析器
      */
     //生成视图解析器,并 为 解析器注入模板引擎
     @Bean
     public ViewResolver viewResolver(SpringTemplateEngine templateEngine){
         ThymeleafViewResolver viewResolver = new ThymeleafViewResolver();
         viewResolver.setCharacterEncoding("UTF-8");
         viewResolver.setTemplateEngine(templateEngine);
         return viewResolver;
     }
 ​
 ​
 ​
     //生成模板引擎 并为模板引擎注入模板解析器
     @Bean
     public SpringTemplateEngine templateEngine(ITemplateResolver templateResolver){
         SpringTemplateEngine templateEngine = new SpringTemplateEngine();
         templateEngine.setTemplateResolver(templateResolver);
         return templateEngine;
     }
 ​
 ​
     //配置生成模板解析器
     @Bean
     public ITemplateResolver templateResolver(){
         WebApplicationContext webApplicationContext = ContextLoader.getCurrentWebApplicationContext();
         ServletContextTemplateResolver templateResolver = new ServletContextTemplateResolver(webApplicationContext.getServletContext());
         templateResolver.setPrefix("/WEB-INF/templates/");
         templateResolver.setSuffix(".html");
         templateResolver.setCharacterEncoding("UTF-8");
         templateResolver.setTemplateMode(TemplateMode.HTML);
         return templateResolver;
 ​
     }
 ​
 ​
 }

Springmvc常用组件

1.springmvc常用组件

  • DispatcherServlet: 前端控制器,不需要工程师开发,由框架提供

    作用:统一处理请求和响应,整个流程控制的中心,由它调用其他组件处理用户的请求

  • HandlerMapping:处理器映射器,不需要工程师开发,由框架提供

    作用:根据请求的url,method等信息查找Handler 即控制器方法

  • Handler: 处理器,需要工程师开发

    作用:在DispatcherServlet的控制下Handler对具体的用户请求进行处理

  • HandlerAdapter: 处理器适配器, 不需要工程师开发, 由框架提供

    作用: 通过HandlerAdapter对处理器(控制器方法)进行执行

  • ViewResolver: 视图解析器,不需要工程师开发, 由框架提供

    作用:进行视图解析,得到相应的视图, 例如 ThymeleafView等

  • View: 视图,

    作用:将模型数据通过页面展示给用户

2.DispatcherServlet初始化过程

DispatcherServlet本质上是一个Servlet,所以天然的遵循Servlet的生命周期,所以宏观上是Servlet生命周期来进行调度

1.初始化WebApplicationContext

所在类:org.springframework.web.servlet.FramewrokServlet

里的 initWebApplicationContext() 方法

2.创建WebApplicationContext

所在类:org.springframework.web.servlet.FramewrokServlet

里的createWebApplicationContext()

3.DispatcherServlet初始化策略

FramewrokServlet创建WebApplicationContext后,刷新容器,

调用 onRefresh()方法此方法在DispatcherServlet中进行了重写

调用了initStrategies()方法,初始化策略,即初始化DispatcherServlet的各个组件

3.DispatcherServlet调用组件处理请求

1.processRequest()

FramewrokServlet重写了HttpServlet中的service() 和doXXX(), 这些方法中调用了processRequest(request,response)

2.doService()

所在类 org.springframework.web.servlet.DispatcherServlet

3.doDispatch() 处理请求响应

所在类 org.springframework.web.servlet.DispatcherServlet

4.processDispatchResult()

所在类 org.springframework.web.servlet.DispatcherServlet

处理模型数据,渲染视图

4.Springmvc执行流程

1.用户向服务器发送请求,请求被springmvc前端控制器DispatcherServlet捕获

2.DispatcherServlet对请求URL进行解析, 得到请求资源标识符(URI),判断请求URI对应的映射:

a. 不存在

i.再判断是否配置了 mvc:default-servlet-handler

ii.如果没配置,则控制台报映射查找不到,客户端展示404错误

No mapping for GET /xx/xx Completed 404 NOT_FOUND

iii.如果有配置,则访问目标资源(一般为静态资源 如 js,css,html), 找不到客户端也会展示 404错误

handler.SimpleHandlerMapping -Mapped to org.springframework.web.servlet.resources.DefaultServletHttpHandler Completed 404 NOT_FOUND

b.存在则执行下面的流程

3.根据该URI,调用HandlerMapping 获得该Handler配置的所有相关的对象(包括Handler对象以及Handler对象对应的拦截器),最后以HandlerExecutionChain执行链对象的形式返回

4.DispatcherServlet根据获得的Handler, 选择一个合适的HandlerAdapter

5.如果成功获得HandlerAdapter,此时将开始执行拦截器的preHandler() 方法 [正向]

6.提取Request中的模型数据,填充Handler入参,开始执行Handler(controller)方法,处理请求,在填充Handler的入参过程中,根据你的配置spring将帮你做一些额外的工作:

a. HttpMessageConverter: 将请求消息(如json,xml等数据)转换成一个对象,将对象转换为指定的响应信息

b.数据转换:对请求消息进行数据转换,如String转换成Integer,Double等

c.数据格式化: 对请求消息进行数据格式化,如将字符串转换成格式化数字或格式化日期等

d.数据验证:验证数据的有效性(长度,格式等),验证结果存储到BindingResult或Error中

7.Handler执行完成后,向DispatcherServlet返回一个ModelAndView对象

8.此时将开始执行拦截器的postHandle() 方法 [逆向]

9.根据返回的ModelAndView(此时会判断是否存在异常,如果存在异常,则执行HandlerExceptionResolver进行异常处理),选择一个适合的ViewResolver进行视图解析,根据Model 和View,来渲染视图

10.渲染视图完毕 ,执行拦截器的afterCompletion()方法[逆向]

11.将渲染结果返回给客户端

  • 29
    点赞
  • 19
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

射手座的程序媛

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值