异常处理器
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 将当前类所标识为 异常处理的组件@ControllerAdvicepublic 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的配置类@Overrideprotected Class<?>[] getRootConfigClasses() {return new Class[]{SpringConfig.class};}//指定springmvc的配置类@Overrideprotected Class<?>[] getServletConfigClasses() {return new Class[]{WebConfig.class};}// 指定dispacherservlet的映射规则@Overrideprotected String[] getServletMappings() {return new String[]{"/"};}//注册 过滤器@Overrideprotected 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个方式 都是 配置 视图解析器*///生成视图解析器,并 为 解析器注入模板引擎@Beanpublic ViewResolver viewResolver(SpringTemplateEngine templateEngine){ThymeleafViewResolver viewResolver = new ThymeleafViewResolver();viewResolver.setCharacterEncoding("UTF-8");viewResolver.setTemplateEngine(templateEngine);return viewResolver;}//生成模板引擎 并为模板引擎注入模板解析器@Beanpublic SpringTemplateEngine templateEngine(ITemplateResolver templateResolver){SpringTemplateEngine templateEngine = new SpringTemplateEngine();templateEngine.setTemplateResolver(templateResolver);return templateEngine;}//配置生成模板解析器@Beanpublic 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配置文件@Configurationpublic class SpringConfig {}@Controllerpublic 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@Overridepublic void addViewControllers(ViewControllerRegistry registry) {registry.addViewController("/hello").setViewName("hello");}//4: default-servlet-handler@Overridepublic void configureDefaultServletHandling(DefaultServletHandlerConfigurer configurer) {configurer.enable();}//8.拦截器@Overridepublic void addInterceptors(InterceptorRegistry registry) {TestInterceptor tt = new TestInterceptor();registry.addInterceptor(tt).addPathPatterns("/**");}//6. 文件上传解析器@Beanpublic MultipartResolver multipartResolver(){CommonsMultipartResolver commonsMultipartResolver =new CommonsMultipartResolver();return commonsMultipartResolver;}//7. 异常处理@Overridepublic 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个方式 都是 配置 视图解析器*///生成视图解析器,并 为 解析器注入模板引擎@Beanpublic ViewResolver viewResolver(SpringTemplateEngine templateEngine){ThymeleafViewResolver viewResolver = new ThymeleafViewResolver();viewResolver.setCharacterEncoding("UTF-8");viewResolver.setTemplateEngine(templateEngine);return viewResolver;}//生成模板引擎 并为模板引擎注入模板解析器@Beanpublic SpringTemplateEngine templateEngine(ITemplateResolver templateResolver){SpringTemplateEngine templateEngine = new SpringTemplateEngine();templateEngine.setTemplateResolver(templateResolver);return templateEngine;}//配置生成模板解析器@Beanpublic 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.将渲染结果返回给客户端