一、拦截器
SpringMVC中内置了拦截器功能,和Servlet的过滤器功能类似,前面我们了解到所有请求的处理的都是由SpringMVC的核心组件:前端控制器,前端控制器由会向三大组件进行转发,由处理单元进行请求处理,视图解析器进行页面渲染,将结果返回浏览器,拦截器就是在前端控制器之后,对请求进行拦截处理
1. 定义拦截器
实现HandlerInterceptor接口即可:
public class MyInterceptor implements HandlerInterceptor {
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
System.out.println("处理单元处理之前");
return true;
}
@Override
public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
System.out.println("处理单元完成处理,响应处理结果之前");
}
@Override
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
System.out.println("视图解析器页面渲染完成,准备返回给浏览器时调用");
}
}
SpringMVC配置文件中注册:
<!--注册拦截器-->
<mvc:interceptors>
<mvc:interceptor>
<mvc:mapping path="/"/>
<bean id="myInterceptor" class="com.aruba.intercept.MyInterceptor"></bean>
</mvc:interceptor>
</mvc:interceptors>
2. 处理分为以下三层
preHandle:
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
System.out.println("处理单元处理之前");
return true;
}
在请求被处理单元处理之前触发,可以通过Servlet请求参数和响应做出处理,返回值为false即拦截请求,不往下分发
postHandle:
@Override
public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
System.out.println("处理单元完成处理,响应处理结果之前");
}
除了preHandle方法的传参之外,还增加了处理单元处理结果:ModelAndView,可以对结果进行修改
afterCompletion:
@Override
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
System.out.println("视图解析器页面渲染完成,准备返回给浏览器时调用");
}
处理最后的收尾工作
二、异常处理
SpringMVC的异常处理是前端控制器交由异常处理器处理的,也就是最终到达Controller层的异常都能被异常处理器捕获,异常处理方式分为以下4种
1. @ExceptionHandler注解
使用@ExceptionHandler注解在Controller层的异常处理方法上
@Controller
public class UserController {
@Autowired
private UserService userService;
@RequestMapping("login.do")
@ResponseBody
public String login(String name) {
User user = userService.findUser(name);
if (user != null) {
return "success";
}
int a = 1 / 0;
return "login failed";
}
@ExceptionHandler
public ModelAndView handlerException(Exception e) {
System.out.println(e.toString());
ModelAndView modelAndView = new ModelAndView();
modelAndView.setViewName("err");
return modelAndView;
}
}
这种方式只能处理该Controller的异常
2. @ControllerAdvice+@ExceptionHandler注解
@ControllerAdvice
public class GloableExceptionHandler {
@ExceptionHandler
public ModelAndView handleException() {
ModelAndView mv = new ModelAndView();
mv.setViewName("err");
return mv;
}
}
该方式异常捕获全局请求
3. SimpleMappingExceptionResolver
该方式使用预置异常处理器,使用xml方式配置:
<!--自定义异常解析器-->
<bean id="exceptionResolver" class="org.springframework.web.servlet.handler.SimpleMappingExceptionResolver">
<property name="exceptionMappings">
<props>
<prop key="java.lang.ArithmeticException">err</prop>
</props>
</property>
</bean>
4. 自定义HandlerExceptionResolver
实现HandlerExceptionResolver接口:
@Configuration
public class MyHandlerExceptionResolver implements HandlerExceptionResolver {
@Override
public ModelAndView resolveException(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Object o, Exception e) {
ModelAndView modelAndView = new ModelAndView();
modelAndView.setViewName("err");
return modelAndView;
}
}
三、其他注解
SpringMVC中还有以下注解:
注解 | 描述 |
---|---|
@PostMapping | 只能是post请求 |
@GetMapping | 只能是get请求 |
@RestController | 表示该Controller层所有请求都使用@ResponseBody注解 |
@JsonFormat | 使用pattern属性指定返回json数据的日期格式,使用timezone属性表示时区 |
@RequestBody | 用于获取json数据字符串,直接注解在实体类请求参数上即可自动转换,required属性可以指定该请求必须含有此请求参数 |
@CrossOrigin | 支持请求跨域,origins属性指定允许访问列表,maxAge属性指定跨域响应前的缓存持续时间 |