系列文章目录
Spring Boot读取配置文件内容的三种方式Spring Boot自动配置–如何切换内置Web服务器SpringBoot项目部署 上述为该系列部分文章,想了解更多可看我博客主页哦!
文章目录
- 系列文章目录
- 前言
- 一、创建自定义过滤器LoginCheckFilter
- 二、在启动类上加入注解@ServletComponentScan
- 三、完善过滤器的处理逻辑
- 3.1 获取本次请求的URI
- 3.2 判断本次请求是否需要处理
- 3.3 如果不需要处理,则直接放行
- 3.4 判断登录状态,如果已登录,则直接放行
- 3.5 未登录情况处理
- 总结
前言
在我们实现完成登录校验功能后,当输入正确的用户名和密码就会跳转到首页,输入错误信息则不跳转首页。但是我们会发现不进行登录,直接在访问路径时直接访问首页的url路径也是可以访问的,就是不用输入用户名和密码也是可以访问首页的。那这样我们的登录功能做的就没有太大的意义。所以我们要进一步去拦截它,使得用户在不登录时访问不了其他的页面。这里就涉及到了今天的过滤器,当然拦截器也是可以实现的,我们这篇文章以过滤器为例来完善登录功能。使得用户在没有登录时访问其他页面时会自动跳回登录页面。
一、创建自定义过滤器LoginCheckFilter
首先要新建一个filter包,在filter包下创建LoginCheckFilter用来实现检查用户是否已经完成登录操作。
在该类上添加WebFilter注解(过滤的注解),里面的参数是filterName就是该过滤器的名字,urlPatterns是需要拦截的路径,这里因为是登录功能,所以我们urlPatterns = "/"表示拦截所有路径(“”通配符),在访问所有路径时都会进入这个函数进行逻辑处理。
然后实现Filter接口下的doFilter方法,将servletRequest、servletResponse强转为HttpServletRequest、HttpServletResponse用来后续做逻辑处理时会用到。
代码如下所示:
/***
* 检查用户是否已经完成登录操作
*/
@WebFilter(filterName = "loginCheckFilter",urlPatterns = "/*")
@Slf4j
public class LoginCheckFilter implements Filter {
@Override
public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
HttpServletRequest request = (HttpServletRequest) servletRequest;
HttpServletResponse response = (HttpServletResponse) servletResponse;
}
}
二、在启动类上加入注解@ServletComponentScan
@ServletComponentScan注解用来扫描你创建的自定义过滤器LoginCheckFilter,所以要在启动类上加上该注解才会生效。
代码如下所示:
@Slf4j
@SpringBootApplication
@ServletComponentScan
public class MyApplication {
public static void main(String[] args) {
SpringApplication.run(MyApplication.class,args);
log.info("项目启动成功...");
}
}
三、完善过滤器的处理逻辑
我们上述两个步骤已经是做完了准备工作,但最核心的如何判断用户是否登录,以及用户在访问什么页面时需要阻止等逻辑代码还没有编写。所以下面我们就来编写逻辑代码。
在filter包下创建的自定义过滤器LoginCheckFilter中编写核心代码。
3.1 获取本次请求的URI
首先我们要获取用户当前访问页面的URI。“/employee/logout”返回的就是这样的路径。
代码如下:
String requestURI = request.getRequestURI();
然后我们要定义出哪些路径是需要放行的,比如用户访问登录页面、静态资源、用户退出等。我们用一个字符串数组存起来。
//直接放行的路径
String[] urls = new String[]{
"/employee/login",
"/employee/logout",
"/backend/**",
"/front/**"
};
3.2 判断本次请求是否需要处理
如果是上述直接放行的路径我们则不需要处理,不是上述字符串数组中的URI我们就需要做进一步的处理。
但我们怎么判断是不是呢?
问题:字符串数组中有"/backend/**"的通配符写法,但我们获取到的URI是固定的写法,例如“/backend/index.html”。这样就不能直接用等号所对比。
解决:根据上述问题,我们就有一个spring自带的用来处理路径匹配的方法AntPathMatcher,可以自动的做上述的匹配操作。
所以我们先初始化代码,如下:
public static final AntPathMatcher PATH_MATCHER=new AntPathMatcher();
编写匹配的逻辑,这里我们将检查是不是需要放行的逻辑放到了一个函数里,如果需要放行就返回true,不放行就返回false(表示字符串数组中的URI没有与之相同的,我们需要拦截)。
代码如下:
public boolean check(String[] urls,String requestURI){
for (String url : urls) {
boolean match = PATH_MATCHER.match(url, requestURI);
if(match){
return true;
}
}
return false;
}
3.3 如果不需要处理,则直接放行
这里就是字符串数组中匹配到与之相对的URI,我们需要放行。
代码如下:
if(check){
filterChain.doFilter(request,response);
return;
}
3.4 判断登录状态,如果已登录,则直接放行
如果用户是已经登录状态,那么当然可以访问首页等页面,所以我们直接放行。就是获取session,查看session中是否有值,null就表示用户没有登录。
代码如下:
if(request.getSession().getAttribute("employee")!=null){
filterChain.doFilter(request,response);
return;
}
3.5 未登录情况处理
如果上述的条件都没有满足,那么就是用户未登录。
则返回未登录结果,通过输出流方式向客户端页面响应数据(因为前端写好了跳转路径,所以我们就只需要返回满足前端代码跳转路径的条件即可。这里为了解释清楚,我们看一下前端的写法。)
前端代码如下:
if (res.data.code === 0 && res.data.msg === 'NOTLOGIN') {// 返回登录页面
console.log('---/backend/page/login/login.html---')
localStorage.removeItem('userInfo')
window.top.location.href = '/backend/page/login/login.html'
} else {
return res.data
}
通过上述的js代码,我们了解到如果满足条件code=0,msg为“NOTLOGIN”则会返回登录页面。所以我们只需要返回一个JSON字符串格式的数据,数据值为code = 0 并且msg ='NOTLOGIN’即可。
java代码如下:
注:R是自定义的返回结果类,返回的数据就包含了上述的code、msg。
response.getWriter().write(JSON.toJSONString(R.error("NOTLOGIN")));
return;
总结
上述就是过滤器完善登录功能的全部代码以及实现流程了,这样我们在未登录时访问首页就会自动跳转至登录页面要求用户登录。
实现起来其实也不算很难,只要记住那些技术要点即可,例如路径匹配AntPathMatcher,加什么注解等。还有就是逻辑处理的部分,将思路理顺后一步一步编写代码,每一小步的代码实现起来也不难,但合起来的整个逻辑需要仔细的查看理解。
这次的文章到这里就结束了,我是心态还需努力呀。我们下篇文章再见!