SpringBoot 實現登錄攔截器(實戰版)

對於管理系統或其他需要用戶登錄的系統,登錄驗證都是必不可少的環節,在 SpringBoot 開發的項目中,通過實現攔截器來實現用戶登錄攔截並驗證。

1、SpringBoot 實現登錄攔截的原理

SpringBoot 通過實現 HandlerInterceptor 接口實現攔截器,通過實現 WebMvcConfigurer 接口實現一個配置類,在配置類中注入攔截器,最後再通過 @Configuration 註解注入配置.

1.1、實現 HandlerInterceptor 接口

實現 HandlerInterceptor 接口需要實現 3 個方法:preHandlepostHandleafterCompletion.

3 個方法各自的功能如下:

package blog.interceptor;

import blog.entity.User;
import org.springframework.web.servlet.HandlerInterceptor;
import org.springframework.web.servlet.ModelAndView;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;

public class UserLoginInterceptor implements HandlerInterceptor {

    /***
     * 在請求處理之前進行調用(Controller方法調用之前)
     */
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        System.out.println("執行了攔截器的preHandle方法");
        try {
            HttpSession session = request.getSession();
            //統一攔截(查詢當前session是否存在user)(這裏user會在每次登錄成功後,寫入session)
            User user = (User) session.getAttribute("user");
            if (user != null) {
                return true;
            }
            response.sendRedirect(request.getContextPath() + "login");
        } catch (Exception e) {
            e.printStackTrace();
        }
        return false;
        //如果設置爲false時,被請求時,攔截器執行到此處將不會繼續操作
        //如果設置爲true時,請求將會繼續執行後面的操作
    }

    /***
     * 請求處理之後進行調用,但是在視圖被渲染之前(Controller方法調用之後)
     */
    @Override
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
        System.out.println("執行了攔截器的postHandle方法");
    }

    /***
     * 整個請求結束之後被調用,也就是在DispatchServlet渲染了對應的視圖之後執行(主要用於進行資源清理工作)
     */
    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
        System.out.println("執行了攔截器的afterCompletion方法");
    }
}

preHandle 在 Controller 之前執行,因此攔截器的功能主要就是在這個部分實現:

1.2、實現 WebMvcConfigurer 接口,註冊攔截器

實現 WebMvcConfigurer 接口來實現一個配置類,將上面實現的攔截器的一個對象註冊到這個配置類中.

package blog.config;

import blog.interceptor.UserLoginInterceptor;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.servlet.config.annotation.InterceptorRegistration;
import org.springframework.web.servlet.config.annotation.InterceptorRegistry;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;

@Configuration
public class LoginConfig implements WebMvcConfigurer {

    @Override
    public void addInterceptors(InterceptorRegistry registry) {
        //註冊TestInterceptor攔截器
        InterceptorRegistration registration = registry.addInterceptor(new UserLoginInterceptor());
        registration.addPathPatterns("/**"); //所有路徑都被攔截
        registration.excludePathPatterns(    //添加不攔截路徑
                "/login",                    //登錄路徑
                "/**/*.html",                //html靜態資源
                "/**/*.js",                  //js靜態資源
                "/**/*.css"                  //css靜態資源
        );
    }
}

將攔截器註冊到了攔截器列表中,並且指明瞭攔截哪些訪問路徑,不攔截哪些訪問路徑,不攔截哪些資源文件;最後再以 @Configuration 註解將配置注入。

1.3、保持登錄狀態

只需一次登錄,如果登錄過,下一次再訪問的時候就無需再次進行登錄攔截,可以直接訪問網站裏面的內容了。

在正確登錄之後,就將 user 保存到 session 中,再次訪問頁面的時候,登錄攔截器就可以找到這個 user 對象,就不需要再次攔截到登錄界面了.

@RequestMapping(value = {"""/""/index"}method = RequestMethod.GET)
public String index(Model model, HttpServletRequest request) {
    User user = (User) request.getSession().getAttribute("user");
    model.addAttribute("user", user);
    return "users/index";
}

@RequestMapping(value = {"/login"}method = RequestMethod.GET)
public String loginIndex() {
    return "users/login";
}

@RequestMapping(value = {"/login"}method = RequestMethod.POST)
public String login(@RequestParam(name = "username")String username, @RequestParam(name = "password")String password,
                    Model model, HttpServletRequest request) {
    User user = userService.getPwdByUsername(username);
    String pwd = user.getPassword();
    String password1 = MD5Utils.md5Code(password).toUpperCase();
    String password2 = MD5Utils.md5Code(password1).toUpperCase();
    if (pwd.equals(password2)) {
        model.addAttribute("user", user);
        request.getSession().setAttribute("user", user);
        return "redirect:/index";
    } else {
        return "users/failed";
    }
}

2、代碼實現及示例

代碼實現如上所示。

在登錄成功之後,將 user 信息保存到 session 中,下一次登錄時瀏覽器根據自己的 SESSIONID 就可以找到對應的 session,就不要再次登錄了,可以從 Chrome 瀏覽器中看到。

3、效果驗證

3.1、訪問 localhost:8081/index 頁面:

被重定向到了 localhost:8081/login,實現了登錄攔截。

3.2、正確輸入用戶名和密碼登錄

3.3、再次訪問 localhost:8081/index

沒有再次被登錄攔截器攔截,證明可以保持登錄。

(感謝閱讀,希望對你所有幫助)

來源:blog.csdn.net/qq_27198345/article/details/111401610

本文由 Readfog 進行 AMP 轉碼,版權歸原作者所有。
來源https://mp.weixin.qq.com/s/HJ6Gfo6nxBKnXpCZqQftLA