欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

最新Spring?Security實戰(zhàn)教程之表單登錄定制到處理邏輯的深度改造(最新推薦)

 更新時間:2025年03月19日 14:45:36   作者:Micro麥可樂  
本章節(jié)介紹了如何通過Spring Security實現(xiàn)從配置自定義登錄頁面、表單登錄處理邏輯的配置,并簡單模擬了前后分離的適配方案,本章節(jié)我們將Spring?Security?默認(rèn)表單進行登錄定制到處理邏輯的深度改造,感興趣的朋友一起看看吧

前言

通過上一章節(jié)《最新Spring Security實戰(zhàn)教程(一)初識Spring Security安全框架》的講解介紹相信大家已經(jīng)認(rèn)識 Spring Security 安全框架,在我們創(chuàng)建第一個項目演示中,相信大家發(fā)現(xiàn)了默認(rèn)表單登錄的局限性
Spring Security 默認(rèn)提供的登錄頁雖然快速可用,但存在三大問題:

  • 界面風(fēng)格與業(yè)務(wù)系統(tǒng)不匹配
  • 登錄成功/失敗處理邏輯固定
  • 缺乏擴展能力(如驗證碼、多因子認(rèn)證)

本章節(jié)我們將Spring Security 默認(rèn)表單進行登錄定制到處理邏輯的深度改造

改造準(zhǔn)備

現(xiàn)在在之前的Maven項目中創(chuàng)建第二個子模塊,命名 login-spring-secutity ,由于我們需要自定義登陸頁,還需要追加引入 thymeleaf 模版框架

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-thymeleaf</artifactId>
</dependency>

完整的maven項目結(jié)構(gòu)如下:

開始登錄頁改造

我們第一步需要自定義自己的帶驗證碼的登陸頁,在 resources/templates 目錄下創(chuàng)建login.html

<!-- src/main/resources/templates/login.html -->
<!DOCTYPE html>
<html xmlns:th="http://www.thymeleaf.org">
<head>
    <meta charset="UTF-8">
    <title>企業(yè)級登錄系統(tǒng)</title>
    <link rel="stylesheet"  rel="external nofollow"  rel="external nofollow"  rel="external nofollow" >
</head>
<body>
<div class="container d-flex justify-content-center align-items-center vh-100">
    <div class="w-100" style="max-width: 400px;">
        <div class="card">
            <div class="card-body">
                <h2 class="card-title text-center mb-4">登錄</h2>
                <form th:action="@{/login}" method="post">
                    <div class="mb-3">
                        <label for="username" class="form-label">用戶名</label>
                        <input type="text" class="form-control" name="username" id="username" placeholder="請輸入用戶名">
                    </div>
                    <div class="mb-3">
                        <label for="password" class="form-label">密碼</label>
                        <input type="password" class="form-control" name="password" id="password" placeholder="請輸入密碼">
                    </div>
                    <div class="d-grid gap-2">
                        <button type="submit" class="btn btn-primary">登錄</button>
                    </div>
                    <p class="mt-3 text-center"><a href="#" rel="external nofollow"  rel="external nofollow" >忘記密碼?</a></p>
                </form>
            </div>
        </div>
    </div>
</div>
</body>
</html>

添加一個默認(rèn)首頁index.html,顯示登出按鈕

<!-- src/main/resources/templates/index.html -->
<html xmlns:th="https://www.thymeleaf.org">
<head>
    <meta charset="UTF-8">
    <title>企業(yè)級登錄系統(tǒng)</title>
    <link rel="stylesheet"  rel="external nofollow"  rel="external nofollow"  rel="external nofollow" >
</head>
<body>
<h1>Hello Security</h1>
<!-- 測試過程不需要關(guān)閉csrf防護 -->
<form th:action="@{/login}" method="post">
    <button type="submit" class="btn btn-primary">Log Out</button>
</form>
<!-- 測試過程需要關(guān)閉csrf防護 否則404 -->
<a th:href="@{/logout}" rel="external nofollow" >Log Out</a>
</body>
</html>

添加 contrller 配置首頁以及登陸頁

@Controller
public class DemoTowController {
    @GetMapping("/login")
    public String login() {
        return "login";
    }
    @GetMapping("/")
    public String index() {
        return "index";
    }
}

最后對 Spring Security 進行配置

@Configuration
public class BasicSecurityConfig {
    // 配置安全策略
    @Bean
    public SecurityFilterChain filterChain(HttpSecurity http) throws Exception {
        http.
                authorizeHttpRequests(authorize -> authorize
                        .anyRequest().authenticated()
                )
                .formLogin(form -> form
                        .loginPage("/login") // 自定義登錄頁路徑
                        .permitAll() //不需要對login認(rèn)證
                )
                .logout(withDefaults())
                .csrf(csrf -> csrf.disable()) //關(guān)閉csrf防護
        ;
        return http.build();
    }
}

測試訪問默認(rèn)訪問主頁,由于主頁被攔截會自動跳轉(zhuǎn)自login登陸頁

輸入正確用戶名密碼后,自動返回主頁,點擊登出按鈕自動回到登錄頁

特別說明:

注意登錄頁以及主頁登出,action 采用 @{} 生成URL,Spring Security會自動幫我們生成name為_csrf 的隱藏表單,作用于 csrf 防護

如果你登出頁是 a 連接形式,為了保證登出不會404的問題
1、我們先關(guān)閉 csrf 防護 http.csrf(csrf -> csrf.disable())
2、登出按鈕使用表單方式 th:action="@{/logout}"

自定義用戶名密碼

到這里有小伙伴又要說了,每次密碼都是Spring Security自動生成的UUID,能自定義用戶名密碼,答案是肯定的。Spring Security提供了在Spring Boot配置文件設(shè)置用戶密碼功能

# 默認(rèn)安全配置(可通過application.yml覆蓋)
spring:
  security:
    user:
      name: admin
      password: admin

登陸成功失敗跳轉(zhuǎn)問題

通過上述代碼,小伙伴們看到登陸成功后,默認(rèn)返回系統(tǒng)主頁 即:index.html頁面,因為業(yè)務(wù)需求需要跳轉(zhuǎn)到別的頁面,如何配置?

Spring Security 配置類中 formLogin 提供了兩個參數(shù) defaultSuccessUrlfailureUrl 方便我們進行配置

http.formLogin(form -> form
    .loginPage("/login") // 自定義登錄頁路徑
    .defaultSuccessUrl("/home", true) // 登錄成功后跳轉(zhuǎn)路徑
    .failureUrl("/login?error=true")  // 登錄失敗后跳轉(zhuǎn)路徑
    .permitAll() //不需要對login認(rèn)證
)

自定義登出

登出和登錄基本相同,由于篇幅問題這里博主就不配置登出的頁面以及登出成功頁面了,主要看以下配置,相信大家都能理解了

http.logout(logout -> logout
    .logoutUrl("/logout") //自定義登出頁
    .logoutSuccessUrl("/login?logout") //登出成功跳轉(zhuǎn)
)

前后端分離適配方案

上述的案例中針對的是前后端都在一個整體中的情況,針對現(xiàn)在前后端分離的項目我們?nèi)绾蝸磉M行改造?我們處理以下問題:

  • 用戶登陸成功返回登陸成功 / 失敗 返回對應(yīng)JSON
  • 用戶登出成功返回登出成功 / 失敗 返回對應(yīng)JSON

這里博主首先引入官方的一個介紹圖,如下:
我們發(fā)現(xiàn)在身份認(rèn)證管理器 AuthenticationManager中, 有兩個結(jié)果 Success 以及 Failure ,最終交給 AuthenticationSuccessHandler 以及 AuthenticationFailureHandler 處理器處理。

簡單總結(jié):

  • 登錄成功調(diào)用:AuthenticationSuccessHandler
  • 登錄失敗調(diào)用:AuthenticationFailureHandler

通過上面的講解,我們只需要自定義這兩個處理器即可,我們在配置文件中增加這兩個處理器,完整代碼如下:

 // 自定義登錄成功處理器
@Configuration
public class BasicSecurityConfig {
    // 配置安全策略
    @Bean
    public SecurityFilterChain filterChain(HttpSecurity http) throws Exception {
        http.
                authorizeHttpRequests(authorize -> authorize
                        .requestMatchers("/ajaxLogin").permitAll() //ajax登陸頁不需要認(rèn)證
                        .anyRequest().authenticated()
                )
                .formLogin(form -> form
                        .loginPage("/login") // 自定義登錄頁路徑
//                        .defaultSuccessUrl("/", true)        // 登錄成功后跳轉(zhuǎn)路徑
//                        .failureUrl("/login?error=true")         // 登錄失敗后跳轉(zhuǎn)路徑
                        .successHandler(loginSuccessHandler())
                        .failureHandler(loginFailureHandler())
                        .permitAll() //不需要對login認(rèn)證
                )
                .logout(withDefaults())
                .csrf(csrf -> csrf.disable()) //關(guān)閉csrf防護
        ;
        return http.build();
    }
    // 自定義登錄成功處理器
    @Bean
    public AuthenticationSuccessHandler loginSuccessHandler() {
        return (request, response, authentication) -> {
            if (isAjaxRequest(request)) {
                response.setContentType("application/json");
                response.setCharacterEncoding("UTF-8");
                response.getWriter().write("{\"code\":200, \"message\":\"/認(rèn)證成功\"}");
            } else {
                response.sendRedirect("/");
            }
        };
    }
    // 自定義登錄失敗處理器
    @Bean
    public AuthenticationFailureHandler loginFailureHandler() {
        return (request, response, exception) -> {
            if (isAjaxRequest(request)) {
                response.setCharacterEncoding("UTF-8");
                response.getWriter().write("{\"code\":401, \"message\":\"認(rèn)證失敗\"}");
            } else {
                response.sendRedirect("/login?error=true");
            }
        };
    }
    //判斷是否ajax請求
    public boolean isAjaxRequest(HttpServletRequest request) {
        String xRequestedWith = request.getHeader("X-Requested-With");
        return "XMLHttpRequest".equals(xRequestedWith);
    }

最后新增一個ajaxLogin.html 使用ajax發(fā)送請求(為了測試方便這里就簡單創(chuàng)建一個,不使用VUE等工程了

<!-- src/main/resources/templates/ajaxLogin.html -->
<!DOCTYPE html>
<html xmlns:th="http://www.thymeleaf.org">
<head>
    <meta charset="UTF-8">
    <title>企業(yè)級登錄系統(tǒng)</title>
    <link rel="stylesheet"  rel="external nofollow"  rel="external nofollow"  rel="external nofollow" >
    <script src="http://libs.baidu.com/jquery/2.0.0/jquery.min.js"></script>
</head>
<body>
<div class="container d-flex justify-content-center align-items-center vh-100">
    <div class="w-100" style="max-width: 400px;">
        <div class="card">
            <div class="card-body">
                <h2 class="card-title text-center mb-4">登錄</h2>
                <form>
                    <div class="mb-3">
                        <label for="username" class="form-label">用戶名</label>
                        <input type="text" class="form-control" name="username" id="username" placeholder="請輸入用戶名">
                    </div>
                    <div class="mb-3">
                        <label for="password" class="form-label">密碼</label>
                        <input type="password" class="form-control" name="password" id="password" placeholder="請輸入密碼">
                    </div>
                    <div class="d-grid gap-2">
                        <button type="submit" class="btn btn-primary">登錄</button>
                    </div>
                    <p class="mt-3 text-center"><a href="#" rel="external nofollow"  rel="external nofollow" >忘記密碼?</a></p>
                </form>
            </div>
        </div>
    </div>
</div>
<script>
    $(document).ready(function () {
        $('form').submit(function (event) {
            event.preventDefault();
            var username = $('#username').val();
            var password = $('#password').val();
            $.ajax({
                type: 'POST',
                url: '/login',
                data: {
                    username: username,
                    password: password
                },
                success: function (response) {
                    console.log(response)
                    if(response.code ==200){
                        window.location.href = '/';
                    }
                }
            })
        })
    })
</script>
</body>
</html>

controller 中追加頁面展示

@GetMapping("/ajaxLogin")
    public String ajaxLogin() {
        return "ajaxLogin";
}

最后啟動Spring Boot項目,訪問 /ajaxLogin 登陸頁,測試輸入正確和不正確的賬號密碼進行測試,并觀察瀏覽器控制臺輸出

結(jié)語

本章節(jié)介紹了如何通過Spring Security實現(xiàn)從配置自定義登錄頁面、表單登錄處理邏輯的配置,并簡單模擬了前后分離的適配方案。小伙伴們可以跟著博主的樣例代碼自己敲一遍進行相關(guān)測試!如果本本章內(nèi)容對您有所幫助,希望 一鍵三連 給博主一點點鼓勵,如果您有任何疑問或建議,請隨時留言討論!

在接下來的章節(jié)中,我們將逐步深入 Spring Security 的各個技術(shù)細節(jié),帶你從入門到精通,全面掌握這一安全技術(shù)的方方面面。歡迎繼續(xù)關(guān)注!

到此這篇關(guān)于最新Spring Security實戰(zhàn)教程之表單登錄定制到處理邏輯的深度改造(最新推薦)的文章就介紹到這了,更多相關(guān)Spring Security表單登錄內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • JavaCV實現(xiàn)讀取視頻信息及自動截取封面圖詳解

    JavaCV實現(xiàn)讀取視頻信息及自動截取封面圖詳解

    javacv可以幫助我們在java中很方便的使用OpenCV以及FFmpeg相關(guān)的功能接口。本文將利用Javacv實現(xiàn)在視頻網(wǎng)站中常見的讀取視頻信息和自動獲取封面圖的功能,感興趣的可以了解一下
    2022-06-06
  • 自定義一個異常類模板的簡單實例

    自定義一個異常類模板的簡單實例

    下面小編就為大家?guī)硪黄远x一個異常類模板的簡單實例。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2016-10-10
  • java?WebSocket?服務(wù)端實現(xiàn)代碼

    java?WebSocket?服務(wù)端實現(xiàn)代碼

    WebSocket協(xié)議是基于TCP的一種新的網(wǎng)絡(luò)協(xié)議。它實現(xiàn)了瀏覽器與服務(wù)器全雙工(full-duplex)通信——允許服務(wù)器主動發(fā)送信息給客戶端,這篇文章主要介紹了java?WebSocket?服務(wù)端代碼,代碼簡單易懂,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2022-02-02
  • Java設(shè)計模式七大原則之接口隔離原則詳解

    Java設(shè)計模式七大原則之接口隔離原則詳解

    接口隔離原則(Interface Segregation Principle),又稱為ISP原則,就是在一個類中不要定義過多的方法,接口應(yīng)該盡量簡單細化。本文將為大家具體介紹一下Java設(shè)計模式七大原則之一的接口隔離原則,需要的可以參考一下
    2022-02-02
  • Apache?Maven3.6.0的下載安裝和環(huán)境配置(圖文教程)

    Apache?Maven3.6.0的下載安裝和環(huán)境配置(圖文教程)

    本文主要介紹了Apache?Maven3.6.0的下載安裝和環(huán)境配置,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2022-07-07
  • Maven重復(fù)依賴問題解決(同一個jar多個版本)

    Maven重復(fù)依賴問題解決(同一個jar多個版本)

    本文主要介紹了Maven重復(fù)依賴問題解決(同一個jar多個版本),文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-06-06
  • idea快速搭建spring cloud注冊中心與注冊的方法

    idea快速搭建spring cloud注冊中心與注冊的方法

    這篇文章主要介紹了idea快速搭建spring cloud注冊中心與注冊的方法,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-07-07
  • Springboot任務(wù)之異步任務(wù)的使用詳解

    Springboot任務(wù)之異步任務(wù)的使用詳解

    今天學(xué)習(xí)了一個新技能SpringBoot實現(xiàn)異步任務(wù),所以特地整理了本篇文章,文中有非常詳細的介紹及代碼示例,需要的朋友可以參考下
    2021-06-06
  • Java實現(xiàn)List反轉(zhuǎn)的方法總結(jié)

    Java實現(xiàn)List反轉(zhuǎn)的方法總結(jié)

    在Java中,反轉(zhuǎn)一個List意味著將其元素的順序顛倒,使得第一個元素變成最后一個,最后一個元素變成第一個,依此類推,這一操作在處理數(shù)據(jù)集合時非常有用,所以本文給大家總結(jié)了Java實現(xiàn)List反轉(zhuǎn)的方法,需要的朋友可以參考下
    2024-04-04
  • 23種設(shè)計模式(9) java橋接模式

    23種設(shè)計模式(9) java橋接模式

    這篇文章主要為大家詳細介紹了java設(shè)計模式之橋接模式,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-11-11

最新評論