最新Spring Security的基于內(nèi)存用戶認(rèn)證方式
1. 前言
又是新的一周,博主繼續(xù)來(lái)給大家更新 Spring Security 實(shí)戰(zhàn)教程了,在上一個(gè)章節(jié)中我們?cè)敿?xì)介紹了 Spring Security 的底層原理,本章節(jié)博主將帶著大家介紹如何在 Spring Security 中實(shí)現(xiàn)基于內(nèi)存的用戶認(rèn)證。
雖然 Spring Security 基于內(nèi)存的用戶認(rèn)證,實(shí)際開(kāi)發(fā)來(lái)說(shuō)相對(duì)來(lái)說(shuō)用的比較少,但某些場(chǎng)景下(如:開(kāi)發(fā)階段、原型驗(yàn)證、演示環(huán)境搭建、單元測(cè)試/集成測(cè)試、或甚至不需要數(shù)據(jù)庫(kù)的簡(jiǎn)單系統(tǒng)),基于內(nèi)存的用戶認(rèn)證方式就足以滿足需求,為了應(yīng)對(duì)這樣需求,博主覺(jué)得還是要必要聊一聊基于內(nèi)存的用戶認(rèn)證。
2. 因何選擇內(nèi)存認(rèn)證?
就如前面說(shuō)的場(chǎng)景,總結(jié)內(nèi)存認(rèn)證主要有以下幾個(gè)優(yōu)點(diǎn):
- 簡(jiǎn)單快捷:配置簡(jiǎn)單,不需要依賴數(shù)據(jù)庫(kù)或外部存儲(chǔ),適用于快速構(gòu)建和測(cè)試。
- 易于調(diào)試:所有用戶信息存儲(chǔ)在代碼中,方便開(kāi)發(fā)過(guò)程中快速定位問(wèn)題。
- 適用于小型應(yīng)用:對(duì)于用戶數(shù)量較少的應(yīng)用或者臨時(shí)驗(yàn)證原型,內(nèi)存認(rèn)證是個(gè)不錯(cuò)的選擇
當(dāng)然,內(nèi)存認(rèn)證也有局限性:用戶數(shù)據(jù)不持久化、無(wú)法擴(kuò)展到分布式系統(tǒng)等。因此,在生產(chǎn)環(huán)境中,通常會(huì)采用基于數(shù)據(jù)庫(kù)或其他外部認(rèn)證機(jī)制的方式。
與數(shù)據(jù)庫(kù)認(rèn)證對(duì)比
| 特性 | 內(nèi)存認(rèn)證 | 數(shù)據(jù)庫(kù)認(rèn)證 |
|---|---|---|
| 用戶存儲(chǔ)位置 | 應(yīng)用內(nèi)存 | 持久化存儲(chǔ) |
| 用戶管理靈活性 | 配置硬編碼 | 動(dòng)態(tài)增刪改查 |
| 生產(chǎn)環(huán)境適用性 | 不推薦 | 推薦 |
3. 基礎(chǔ)配置實(shí)戰(zhàn)
接下來(lái)在之前的Maven項(xiàng)目中還是創(chuàng)建第三個(gè)子模塊 memory-spring-security ,完整的maven項(xiàng)目結(jié)構(gòu)如下:

? 創(chuàng)建Spring Security配置文件
現(xiàn)在我們來(lái)創(chuàng)建一個(gè)Spring Security 配置文件 InMemorySecurityConfig
@Configuration
public class InMemorySecurityConfig {
// 手動(dòng)配置用戶信息
@Bean
public UserDetailsService users() {
UserDetails user = User.withUsername("user")
.password("{noop}user") // {noop}表示不加密
.roles("USER")
.build();
UserDetails admin = User.withUsername("admin")
.password("{noop}admin")
.roles("ADMIN")
.build();
//可以繼續(xù)追加其它用戶...
UserDetails anonymous = User.withUsername("anonymous")
.password("{noop}anonymous")
.roles("ANONYMOUS")
.build();
return new InMemoryUserDetailsManager(user, admin, anonymous);
}
// 配置安全策略 并配置/admin/** 只允許ADMIN角色用戶訪問(wèn)
@Bean
public SecurityFilterChain filterChain(HttpSecurity http) throws Exception {
http.
authorizeHttpRequests(authorize -> authorize
.requestMatchers("/admin/**").hasRole("ADMIN")
.anyRequest().authenticated()
)
.formLogin(withDefaults())
.logout(withDefaults())
;
return http.build();
}
}配置說(shuō)明
- 構(gòu)建UserDetailsService
創(chuàng)建兩個(gè)用戶信息分別是:admin、user ,并由InMemoryUserDetailsManager進(jìn)行在內(nèi)存中保存用戶數(shù)據(jù)- SecurityFilterChain
在SecurityFilterChain中,默認(rèn)采用了Spring Security表單登陸登出方式,并配置/admin/**請(qǐng)求路徑下需要管理員角色方可訪問(wèn)
? 創(chuàng)建Controller測(cè)試
接下來(lái)我們創(chuàng)建用以測(cè)試的Controller :DemoMemoryController
@Controller
public class DemoMemoryController {
//返回用戶信息及角色權(quán)限
@GetMapping("/")
public ResponseEntity<Map<String, Object>> index(Authentication authentication) {
String username = authentication.getName();//用戶名
Object principal =authentication.getPrincipal();//身份
// 獲取用戶擁有的權(quán)限列表
List<String> roles = authentication.getAuthorities().stream()
.map(GrantedAuthority::getAuthority)
.collect(Collectors.toList());
//返回用戶信息
return ResponseEntity.ok(Map.of(
"username", username,
"principal", principal,
"roles", roles));
}
//測(cè)試管理員權(quán)限
@GetMapping("/admin/view")
public ResponseEntity<String> admin() {
return ResponseEntity.ok("管理員ADMIN角色訪問(wèn)ok");
}
}代碼說(shuō)明
- 注入 Authentication 對(duì)象
在index方法中,通過(guò)方法參數(shù)直接注入Authentication對(duì)象,Spring Security會(huì)自動(dòng)傳入當(dāng)前認(rèn)證信息。- 提取用戶信息
通過(guò)authentication.getName()獲取當(dāng)前登錄用戶的用戶名;通過(guò)authentication.getAuthorities()獲取用戶的權(quán)限列表,并將每個(gè)權(quán)限的 getAuthority() 值收集成一個(gè)字符串列表。
? 運(yùn)行測(cè)試
啟動(dòng)項(xiàng)目訪問(wèn),登陸頁(yè)中分別測(cè)試兩個(gè)用戶登陸查看信息,如user用戶:

接下來(lái)嘗試使用user用戶訪問(wèn) /admin/view 路徑,會(huì)出現(xiàn) 403 訪問(wèn)錯(cuò)誤提示:即您無(wú)權(quán)訪問(wèn)此地址
切換admin用戶登陸,繼續(xù)訪問(wèn) /admin/view 路徑,出現(xiàn) 管理員ADMIN角色訪問(wèn)ok 文字顯示即代表管理員角色允許訪問(wèn)
4. 追加密碼編碼器
在上述代碼中,我們使用了 password(“{noop}admin”) 聲明了密碼明文存儲(chǔ),如果我們需要對(duì)密碼加密,如何操作? 實(shí)際上 Spring Security 為我們提供了非常方便的密碼編碼器
密碼編碼器配置
只需要?jiǎng)?chuàng)建 PasswordEncoder Bean,并返回加密類型,如下代碼樣例:
@Bean
public PasswordEncoder passwordEncoder() {
return new BCryptPasswordEncoder();
}
@Bean
public UserDetailsService users(PasswordEncoder encoder) {
UserDetails user = User.builder()
.username("user")
.password(encoder.encode("secret"))
.roles("USER")
.build();
return new InMemoryUserDetailsManager(user);
}支持的編碼格式:
# 不同前綴對(duì)應(yīng)不同編碼器
{noop} → NoOpPasswordEncoder (明文)
{bcrypt} → BCryptPasswordEncoder
{pbkdf2} → Pbkdf2PasswordEncoder
{scrypt} → SCryptPasswordEncoder
{sha256} → StandardPasswordEncoder
5. 總結(jié)
通過(guò)本章節(jié)的配置示例,相信你已經(jīng)可以使用 Spring Security 的基于內(nèi)存認(rèn)證方式來(lái)快速搭建安全認(rèn)證體系。
注意章節(jié)中提到的基于內(nèi)存的用戶認(rèn)證適用的場(chǎng)景,更多情況下還是建議使用 更為健全的認(rèn)證方式,如 基于數(shù)據(jù)庫(kù)的認(rèn)證、JWT 令牌認(rèn)證或 OAuth2,在下一章節(jié)我們將重點(diǎn)講述數(shù)據(jù)庫(kù)的認(rèn)證,敬請(qǐng)期待…
到此這篇關(guān)于最新Spring Security的基于內(nèi)存用戶認(rèn)證方式的文章就介紹到這了,更多相關(guān)Spring Security內(nèi)存用戶認(rèn)證內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- SpringBoot+Spring Security基于內(nèi)存用戶認(rèn)證的實(shí)現(xiàn)
- Spring Security內(nèi)存中認(rèn)證的實(shí)現(xiàn)
- SpringSecurity+Redis+Jwt實(shí)現(xiàn)用戶認(rèn)證授權(quán)
- Spring Security實(shí)現(xiàn)登錄認(rèn)證實(shí)戰(zhàn)教程
- SpringSecurity進(jìn)行認(rèn)證與授權(quán)的示例代碼
- springsecurity實(shí)現(xiàn)用戶登錄認(rèn)證快速使用示例代碼(前后端分離項(xiàng)目)
相關(guān)文章
Java?多個(gè)時(shí)間區(qū)間進(jìn)行合并處理方法
用戶在選擇多個(gè)時(shí)間區(qū)間之后,如選擇的時(shí)間區(qū)間連續(xù)或者有重疊,需要對(duì)所選的時(shí)間區(qū)間進(jìn)行合并,這其實(shí)是一個(gè)區(qū)間合并問(wèn)題,下面通過(guò)本文給大家介紹Java?多個(gè)時(shí)間區(qū)間進(jìn)行合并處理的解決方案,一起看看吧2024-02-02
SpringBoot Web詳解靜態(tài)資源規(guī)則與定制化處理
這篇文章主要介紹了SpringBoot web場(chǎng)景的靜態(tài)資源規(guī)則與定制化,文章圍繞主題展開(kāi)詳細(xì)的內(nèi)容介紹,具有一定的參考價(jià)值,需要的小伙伴可以參考一下2022-06-06
SpringBoot @SpringBootTest加速單元測(cè)試的小訣竅
這篇文章主要介紹了SpringBoot @SpringBootTest加速單元測(cè)試的小訣竅,具有很好的參考價(jià)值,對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-11-11
SpringBoot開(kāi)發(fā)中使用DTO層的方法示例
DTO層是在應(yīng)用程序的業(yè)務(wù)邏輯層和數(shù)據(jù)訪問(wèn)層之間引入的一個(gè)中間層,用于在不同層之間傳輸數(shù)據(jù),本文主要介紹了SpringBoot開(kāi)發(fā)中使用DTO層,具有一定的參考價(jià)值,感興趣的可以了解一下2024-06-06
透徹理解Java中Synchronized(對(duì)象鎖)和Static Synchronized(類鎖)的區(qū)別
這篇文章主要介紹了Java中Synchronized(對(duì)象鎖)和Static Synchronized(類鎖)的區(qū)別,希望對(duì)大家有所幫助,一起跟隨小編過(guò)來(lái)看看吧2018-05-05
Java實(shí)現(xiàn)的文件上傳下載工具類完整實(shí)例【上傳文件自動(dòng)命名】
這篇文章主要介紹了Java實(shí)現(xiàn)的文件上傳下載工具類,結(jié)合完整實(shí)例形式分析了java針對(duì)文件上傳下載操作的相關(guān)實(shí)現(xiàn)技巧,并且針對(duì)上傳文件提供了自動(dòng)命名功能以避免文件命名重復(fù),需要的朋友可以參考下2017-11-11
詳解spring cloud Feign使用中遇到的問(wèn)題總結(jié)
本篇文章主要介紹了詳解spring cloud Feign使用中遇到的問(wèn)題總結(jié),小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-01-01
Java將不同的List集合復(fù)制到另一個(gè)集合常見(jiàn)的方法
在Java中,有時(shí)候我們需要將一個(gè)List對(duì)象的屬性值復(fù)制到另一個(gè)List對(duì)象中,使得兩個(gè)對(duì)象的屬性值相同,這篇文章主要介紹了Java將不同的List集合復(fù)制到另一個(gè)集合常見(jiàn)的方法,需要的朋友可以參考下2024-09-09
Java實(shí)現(xiàn)pdf轉(zhuǎn)圖片案例
這篇文章主要介紹了Java實(shí)現(xiàn)pdf轉(zhuǎn)圖片案例,按照步驟放置代碼,一步步完成該案例,將代碼部署便可,需要的朋友可以參考下2021-06-06

