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
去掉或注释掉 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
使用配置类和注解代替 web.xml 和springmvc配置文件的功能
在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
?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;
??
? ? }
??
??
?}
DispatcherServlet: 前端控制器,不需要工程师开发,由框架提供
作用:统一处理请求和响应,整个流程控制的中心,由它调用其他组件处理用户的请求
HandlerMapping:处理器映射器,不需要工程师开发,由框架提供
作用:根据请求的url,method等信息查找Handler 即控制器方法
Handler: 处理器,需要工程师开发
作用:在DispatcherServlet的控制下Handler对具体的用户请求进行处理
HandlerAdapter: 处理器适配器, 不需要工程师开发, 由框架提供
作用: 通过HandlerAdapter对处理器(控制器方法)进行执行
ViewResolver: 视图解析器,不需要工程师开发, 由框架提供
作用:进行视图解析,得到相应的视图, 例如 ThymeleafView等
View: 视图,
作用:将模型数据通过页面展示给用户
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的各个组件
FramewrokServlet重写了HttpServlet中的service() 和doXXX(), 这些方法中调用了processRequest(request,response)
所在类 org.springframework.web.servlet.DispatcherServlet
所在类 org.springframework.web.servlet.DispatcherServlet
所在类 org.springframework.web.servlet.DispatcherServlet
处理模型数据,渲染视图
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.将渲染结果返回给客户端