SpringBoot起飞系列-拦截器和统一错误处理(七)

一、前言


在前边部分我们已经学会了基本的web开发流程,在web开发中,我们通常会对请求做统一处理,比如未登录的用户要拦截掉相关请求,报错页面统一显示等等,这些都需要配置,可以大大简化我们的代码,实现功能的完整性与统一性。


二、拦截器配置


首先我们先做一个登录身份验证拦截器,来拦截那些没有登录的用户,保护我们的资源。下面我们创建一个拦截器,需要实现拦截器接口。



 1 package com.example.demo.component;
2
3 import org.springframework.web.servlet.HandlerInterceptor;
4 import org.springframework.web.servlet.ModelAndView;
5
6 import javax.servlet.http.HttpServletRequest;
7 import javax.servlet.http.HttpServletResponse;
8
9 public class LoginHandlerInterceptor implements HandlerInterceptor {
10
11 @Override
12 public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
13 if(request.getSession().getAttribute('loginUser') == null){
14 request.setAttribute('msg','请先登录');
15 request.getRequestDispatcher('/index.html').forward(request,response);
16 return false;
17 }
18 return true;
19 }
20
21 @Override
22 public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
23
24 }
25
26 @Override
27 public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
28
29 }
30 }


这个拦截器的作用是获取当前session中的loginUser属性,如果有值说明登录了就可以放行,当然我们需要在登录成功的代码里边设置好session的这个属性。


在WebConfig中添加我们的拦截器:



 1     //所有的WebMvcConfigurerAdapter组件都会一起起作用
2 @Bean //将组件注册在容器
3 public WebMvcConfigurerAdapter webMvcConfigurerAdapter(){
4 WebMvcConfigurerAdapter adapter = new WebMvcConfigurerAdapter() {
5 @Override
6 public void addViewControllers(ViewControllerRegistry registry) {
7 registry.addViewController('/').setViewName('login');
8 registry.addViewController('/index.html').setViewName('login');
9 }
10
11 @Override
12 public void addInterceptors(InterceptorRegistry registry) {
13 registry.addInterceptor(new LoginHandlerInterceptor())
14 .excludePathPatterns('/index.html','/','/user/login');
15 }
16 };
17 return adapter;
18 }


第12-15行,注册我们拦截器之后,排除掉不需要验证的页面,默认情况下静态资源不会做验证。


三、统一错误处理


springboot默认已经配置了统一错误处理,只不过错误页面是内置的,可能不是我们想要的,所以如果我们要自定义错误页面,还得重新配置。


3.1 定制错误页面


在使用模板的情况下,springboot默认会找到/templates/error/xxx.html页面作为错误页面显示,比如我们用4xx.html处理所有httpstatuscode以4开头的错误,比如401,403,404,如果有具体的数字,就先用具体的数字对应页面,如果没有就用有前缀开头的页面。


显示错误页面时,页面上能获取到以下信息,可以让我们看到详细的错误信息:


timestamp:时间戳。


status:状态码。


error:错误提示。


exception:异常对象。


message:异常消息。


errors:JSR303数据校验的错误都在这里。


在没有模板的情况下,会去找静态资源下的相关页面,如果静态资源下也没有,就用springboot默认的错误页面。如下,我添加了一个4xx.html的错误处理页面,当我访问一个不存在的路径(404错误)时,就会显示我添加的4xx.html页面:




3.2 自定义返回的json数据


默认情况下springboot会出现自适应的错误显示,当用浏览器访问(接受类型为text/html)时会显示错误页面,当用postman(接受类型为application/json)会显示错误json数据。不过显示的字段都是内置固定的,如果我们想要添加自己的错误数据,就要自己定制了。


在springboot中,出现错误以后,会来到/error请求,会被BasicErrorController处理,响应出去可以获取的数据是由getErrorAttributes得到的(是AbstractErrorController(ErrorController)规定的方法),这里我们就可以编写一个ErrorController的实现类【或者是编写AbstractErrorController的子类】,放在容器中替换掉原来的ErrorController,页面上能用的数据,或者是json返回能用的数据都是通过errorAttributes.getErrorAttributes得到。


添加自定一个ErrorAttribute,重写errorAttributes.getErrorAttributes,返回自己的数据map:



 1 package com.example.demo.component;
2
3 import org.springframework.boot.web.servlet.error.DefaultErrorAttributes;
4 import org.springframework.stereotype.Component;
5 import org.springframework.web.context.request.WebRequest;
6
7 import java.util.Map;
8
9 @Component
10 public class MyErrorAttributes extends DefaultErrorAttributes {
11 @Override
12 public Map<String, Object> getErrorAttributes(WebRequest webRequest, boolean includeStackTrace) {
13 Map<String, Object> map = super.getErrorAttributes(webRequest, includeStackTrace);
14 map.put('code',1);
15 map.put('msg','自定义错误');
16 return map;
17 }
18 }


访问错误页面和json返回:




四、总结


不得不说,springboot真是很懂我们,把所有东西都配置好了,只要我们稍微修修补补就能很好的满足我们的需求了,这里我们可以完全不要原始的错误信息,可以把错误信息获取数据组装成我们标准统一格式的错误信息,也可以在这里统一记录我们的错误日志,也是非常的方便。

原文地址:https://www.cnblogs.com/Yangbuyi/p/13388325.html