springmvc提供了一个处理控制器方法执行过程中所出现的异常的接口: HandlerExceptionResolver
HandlerExceptionResolver接口的实现类有: DefaultHandlerExceptionResolver和SimpleMappingExceptionResolver
springmvc提供了自定义的异常处理器SimpleMappingExceptionResolver, 使用方式: (注意拦截器要放行)
error
@RequestMapping("/testExp")
public String testExp(){
System.out.println(1/0);
return "success";
}
编写 error.html
出现错误 显示异常错误信息
页面显示:
出现错误
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
首页
启动项目 即可访问 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 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.将渲染结果返回给客户端