日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術(shù)文章
文章詳情頁

Spring Boot 中密碼加密的兩種方法

瀏覽:10日期:2023-08-17 08:48:45

先說一句:密碼是無法解密的。大家也不要再問松哥微人事項(xiàng)目中的密碼怎么解密了!

密碼無法解密,還是為了確保系統(tǒng)安全。今天松哥就來和大家聊一聊,密碼要如何處理,才能在最大程度上確保我們的系統(tǒng)安全。

1.為什么要加密

2011 年 12 月 21 日,有人在網(wǎng)絡(luò)上公開了一個(gè)包含 600 萬個(gè) CSDN 用戶資料的數(shù)據(jù)庫,數(shù)據(jù)全部為明文儲(chǔ)存,包含用戶名、密碼以及注冊郵箱。事件發(fā)生后 CSDN 在微博、官方網(wǎng)站等渠道發(fā)出了聲明,解釋說此數(shù)據(jù)庫系 2009 年備份所用,因不明原因泄露,已經(jīng)向警方報(bào)案,后又在官網(wǎng)發(fā)出了公開道歉信。在接下來的十多天里,金山、網(wǎng)易、京東、當(dāng)當(dāng)、新浪等多家公司被卷入到這次事件中。整個(gè)事件中最觸目驚心的莫過于 CSDN 把用戶密碼明文存儲(chǔ),由于很多用戶是多個(gè)網(wǎng)站共用一個(gè)密碼,因此一個(gè)網(wǎng)站密碼泄露就會(huì)造成很大的安全隱患。由于有了這么多前車之鑒,我們現(xiàn)在做系統(tǒng)時(shí),密碼都要加密處理。

這次泄密,也留下了一些有趣的事情,特別是對于廣大程序員設(shè)置密碼這一項(xiàng)。人們從 CSDN 泄密的文件中,發(fā)現(xiàn)了一些好玩的密碼,例如如下這些:

ppnn13%dkstFeb.1st 這段密碼的中文解析是:娉娉裊裊十三余,豆蔻梢頭二月初。 csbt34.ydhl12s 這段密碼的中文解析是:池上碧苔三四點(diǎn),葉底黃鸝一兩聲 ...

等等不一而足,你會(huì)發(fā)現(xiàn)很多程序員的人文素養(yǎng)還是非常高的,讓人嘖嘖稱奇。

2.加密方案

密碼加密我們一般會(huì)用到散列函數(shù),又稱散列算法、哈希函數(shù),這是一種從任何數(shù)據(jù)中創(chuàng)建數(shù)字“指紋”的方法。

散列函數(shù)把消息或數(shù)據(jù)壓縮成摘要,使得數(shù)據(jù)量變小,將數(shù)據(jù)的格式固定下來,然后將數(shù)據(jù)打亂混合,重新創(chuàng)建一個(gè)散列值。散列值通常用一個(gè)短的隨機(jī)字母和數(shù)字組成的字符串來代表。好的散列函數(shù)在輸入域中很少出現(xiàn)散列沖突。在散列表和數(shù)據(jù)處理中,不抑制沖突來區(qū)別數(shù)據(jù),會(huì)使得數(shù)據(jù)庫記錄更難找到。

我們常用的散列函數(shù)有 MD5 消息摘要算法、安全散列算法(Secure Hash Algorithm)。

但是僅僅使用散列函數(shù)還不夠,單純的只使用散列函數(shù),如果兩個(gè)用戶密碼明文相同,生成的密文也會(huì)相同,這樣就增加的密碼泄漏的風(fēng)險(xiǎn)。

為了增加密碼的安全性,一般在密碼加密過程中還需要加鹽,所謂的鹽可以是一個(gè)隨機(jī)數(shù)也可以是用戶名,加鹽之后,即使密碼明文相同的用戶生成的密碼密文也不相同,這可以極大的提高密碼的安全性。

傳統(tǒng)的加鹽方式需要在數(shù)據(jù)庫中有專門的字段來記錄鹽值,這個(gè)字段可能是用戶名字段(因?yàn)橛脩裘ㄒ唬部赡苁且粋€(gè)專門記錄鹽值的字段,這樣的配置比較繁瑣。

Spring Security 提供了多種密碼加密方案,官方推薦使用 BCryptPasswordEncoder,BCryptPasswordEncoder 使用 BCrypt 強(qiáng)哈希函數(shù),開發(fā)者在使用時(shí)可以選擇提供 strength 和 SecureRandom 實(shí)例。strength 越大,密鑰的迭代次數(shù)越多,密鑰迭代次數(shù)為 2^strength。strength 取值在 4~31 之間,默認(rèn)為 10。

不同于 Shiro 中需要自己處理密碼加鹽,在 Spring Security 中,BCryptPasswordEncoder 就自帶了鹽,處理起來非常方便。

3.實(shí)踐

3.1 codec 加密

commons-codec 是一個(gè) Apache 上的開源項(xiàng)目,用它可以方便的實(shí)現(xiàn)密碼加密。松哥在 V 部落 項(xiàng)目中就是采用的這種方案(https://github.com/lenve/VBlog)。在 Spring Security 還未推出 BCryptPasswordEncoder 的時(shí)候,commons-codec 還是一個(gè)比較常見的解決方案。

所以,這里我先來給大家介紹下 commons-codec 的用法。

首先我們需要引入 commons-codec 的依賴:

<dependency> <groupId>commons-codec</groupId> <artifactId>commons-codec</artifactId> <version>1.11</version></dependency>

然后自定義一個(gè) PasswordEncoder:

@Componentpublic class MyPasswordEncoder implements PasswordEncoder { @Override public String encode(CharSequence rawPassword) { return DigestUtils.md5DigestAsHex(rawPassword.toString().getBytes()); } @Override public boolean matches(CharSequence rawPassword, String encodedPassword) { return encodedPassword.equals(DigestUtils.md5DigestAsHex(rawPassword.toString().getBytes())); }}

在 Spring Security 中,PasswordEncoder 專門用來處理密碼的加密與比對工作,我們自定義 MyPasswordEncoder 并實(shí)現(xiàn) PasswordEncoder 接口,還需要實(shí)現(xiàn)該接口中的兩個(gè)方法:

encode 方法表示對密碼進(jìn)行加密,參數(shù) rawPassword 就是你傳入的明文密碼,返回的則是加密之后的密文,這里的加密方案采用了 MD5。 matches 方法表示對密碼進(jìn)行比對,參數(shù) rawPassword 相當(dāng)于是用戶登錄時(shí)傳入的密碼,encodedPassword 則相當(dāng)于是加密后的密碼(從數(shù)據(jù)庫中查詢而來)。

最后記得將 MyPasswordEncoder 通過 @Component 注解標(biāo)記為 Spring 容器中的一個(gè)組件。

這樣用戶在登錄時(shí),就會(huì)自動(dòng)調(diào)用 matches 方法進(jìn)行密碼比對。

當(dāng)然,使用了 MyPasswordEncoder 之后,在用戶注冊時(shí),就需要將密碼加密之后存入數(shù)據(jù)庫中,方式如下:

public int reg(User user) { ... //插入用戶,插入之前先對密碼進(jìn)行加密 user.setPassword(passwordEncoder.encode(user.getPassword())); result = userMapper.reg(user); ...}

其實(shí)很簡單,就是調(diào)用 encode 方法對密碼進(jìn)行加密。完整代碼大家可以參考 V 部落(https://github.com/lenve/VBlog),我這里就不贅述了。

3.2 BCryptPasswordEncoder 加密

但是自己定義 PasswordEncoder 還是有些麻煩,特別是處理密碼加鹽問題的時(shí)候。

所以在 Spring Security 中提供了 BCryptPasswordEncoder,使得密碼加密加鹽變得非常容易。只需要提供 BCryptPasswordEncoder 這個(gè) Bean 的實(shí)例即可,微人事就是采用了這種方案(https://github.com/lenve/vhr),如下:

@BeanPasswordEncoder passwordEncoder() { return new BCryptPasswordEncoder(10);}

創(chuàng)建 BCryptPasswordEncoder 時(shí)傳入的參數(shù) 10 就是 strength,即密鑰的迭代次數(shù)(也可以不配置,默認(rèn)為 10)。同時(shí),配置的內(nèi)存用戶的密碼也不再是 123 了,如下:

auth.inMemoryAuthentication().withUser('admin').password('$2a$10$RMuFXGQ5AtH4wOvkUqyvuecpqUSeoxZYqilXzbz50dceRsga.WYiq').roles('ADMIN', 'USER').and().withUser('sang').password('$2a$10$eUHbAOMq4bpxTvOVz33LIehLe3fu6NwqC9tdOcxJXEhyZ4simqXTC').roles('USER');

這里的密碼就是使用 BCryptPasswordEncoder 加密后的密碼,雖然 admin 和 sang 加密后的密碼不一樣,但是明文都是 123。配置完成后,使用 admin/123 或者 sang/123 就可以實(shí)現(xiàn)登錄。

本案例使用了配置在內(nèi)存中的用戶,一般情況下,用戶信息是存儲(chǔ)在數(shù)據(jù)庫中的,因此需要在用戶注冊時(shí)對密碼進(jìn)行加密處理,如下:

@Servicepublic class RegService { public int reg(String username, String password) { BCryptPasswordEncoder encoder = new BCryptPasswordEncoder(10); String encodePasswod = encoder.encode(password); return saveToDb(username, encodePasswod); }}

用戶將密碼從前端傳來之后,通過調(diào)用 BCryptPasswordEncoder 實(shí)例中的 encode 方法對密碼進(jìn)行加密處理,加密完成后將密文存入數(shù)據(jù)庫。

4.源碼淺析

最后我們再來稍微看一下 PasswordEncoder。

PasswordEncoder 是一個(gè)接口,里邊只有三個(gè)方法:

public interface PasswordEncoder { String encode(CharSequence rawPassword); boolean matches(CharSequence rawPassword, String encodedPassword); default boolean upgradeEncoding(String encodedPassword) { return false; }} encode 方法用來對密碼進(jìn)行加密。 matches 方法用來對密碼進(jìn)行比對。 upgradeEncoding 表示是否需要對密碼進(jìn)行再次加密以使得密碼更加安全,默認(rèn)為 false。

Spring Security 為 PasswordEncoder 提供了很多實(shí)現(xiàn):

Spring Boot 中密碼加密的兩種方法

但是老實(shí)說,自從有了 BCryptPasswordEncoder,我們很少關(guān)注其他實(shí)現(xiàn)類了。

PasswordEncoder 中的 encode 方法,是我們在用戶注冊的時(shí)候手動(dòng)調(diào)用。

matches 方法,則是由系統(tǒng)調(diào)用,默認(rèn)是在 DaoAuthenticationProvider#additionalAuthenticationChecks 方法中調(diào)用的。

protected void additionalAuthenticationChecks(UserDetails userDetails, UsernamePasswordAuthenticationToken authentication) throws AuthenticationException { if (authentication.getCredentials() == null) { logger.debug('Authentication failed: no credentials provided'); throw new BadCredentialsException(messages.getMessage( 'AbstractUserDetailsAuthenticationProvider.badCredentials', 'Bad credentials')); } String presentedPassword = authentication.getCredentials().toString(); if (!passwordEncoder.matches(presentedPassword, userDetails.getPassword())) { logger.debug('Authentication failed: password does not match stored value'); throw new BadCredentialsException(messages.getMessage( 'AbstractUserDetailsAuthenticationProvider.badCredentials', 'Bad credentials')); }}

可以看到,密碼比對就是通過 passwordEncoder.matches 方法來進(jìn)行的。

關(guān)于 DaoAuthenticationProvider 的調(diào)用流程,大家可以參考 SpringSecurity 自定義認(rèn)證邏輯的兩種方式(高級玩法)一文。

好了,今天就和小伙伴們簡單聊一聊 Spring Security 加密問題,小伙伴們要是有收獲記得點(diǎn)個(gè)在看鼓勵(lì)下松哥哦~

以上就是Spring Boot 中密碼加密的兩種方法的詳細(xì)內(nèi)容,更多關(guān)于Spring Boot 密碼加密的資料請關(guān)注好吧啦網(wǎng)其它相關(guān)文章!

標(biāo)簽: Spring
相關(guān)文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
国产精品久久久久久久免费软件| 久久一级电影| 亚洲一区二区三区四区五区午夜| 免费在线欧美黄色| 亚洲欧美久久久| 蜜芽一区二区三区| 亚洲精品在线国产| 欧美在线看片| 久久精品国产网站| 美女精品视频在线| 久久av偷拍| 成人台湾亚洲精品一区二区| 麻豆成全视频免费观看在线看| 欧美国产美女| 日韩国产网站| 免费视频国产一区| 一本一本久久| 亚洲另类黄色| 国产精品试看| 中文精品电影| 欧美日一区二区在线观看| 久久xxx视频| 国产999精品在线观看| 久久精品亚洲欧美日韩精品中文字幕| 日韩精品一区二区三区免费观看| 日韩视频在线一区二区三区| 国产中文字幕一区二区三区| 日韩不卡免费高清视频| 欧美精品羞羞答答| 亚洲精品精选| 国产精品magnet| 日韩精品首页| 中文字幕一区二区三区四区久久| 免费欧美在线视频| 久久国内精品视频| 日本久久黄色| 在线亚洲国产精品网站| 日韩av资源网| 亚洲最新无码中文字幕久久 | 影院欧美亚洲| 日韩欧美激情电影| 国产一区二区三区天码| 精品91久久久久| 国产精品一区高清| 国产中文字幕一区二区三区| 91高清一区| 欧美片网站免费| 亚洲精品在线影院| 婷婷亚洲成人| 精品国产一区二区三区av片| 女同性一区二区三区人了人一 | 丝袜美腿一区| 午夜性色一区二区三区免费视频| 九九久久国产| 亚洲精品2区| 你懂的亚洲视频| 久久久成人网| 国产一卡不卡| 亚洲激情精品| 久久精品国产999大香线蕉| 五月天激情综合网| 欧美日韩亚洲一区在线观看| 欧美亚洲国产一区| 国产精品天天看天天狠| 国产专区一区| 欧美国产极品| 美女被久久久| 久久久久久色| 99精品在线观看| 91亚洲精品视频在线观看 | 亚洲91视频| 日韩国产在线观看一区| 久久久精品五月天| 国产欧美日韩免费观看| 一区三区视频| 亚洲女同av| 国产精品一区免费在线| 亚洲一区观看| 亚洲成人国产| 美女毛片一区二区三区四区最新中文字幕亚洲| 在线精品小视频| 日韩国产一区| 精品成人18| 奇米777国产一区国产二区| 日韩精品欧美激情一区二区| 国产精品igao视频网网址不卡日韩 | 日韩a一区二区| 青草av.久久免费一区| 亚洲欧美久久| 日韩精品欧美激情一区二区| 国产黄色一区| 视频一区日韩精品| 亚洲精品一区二区在线看| 麻豆视频在线观看免费网站黄| 天堂va在线高清一区| 亚洲欧洲一区二区天堂久久| 日韩免费在线| 精品99在线| 国产精品第一| 亚洲精品乱码久久久久久蜜桃麻豆 | 欧美日韩91| 欧美.日韩.国产.一区.二区| 国产成人精品福利| 久久av免费| 国产精品一区二区三区www| 亚洲精品欧美| 亚洲国产专区校园欧美| 国产夫妻在线| 国产+成+人+亚洲欧洲在线| 国产精品久久久久久久免费软件| 日韩三级久久| 日韩一二三区在线观看| 免费中文字幕日韩欧美| 99热免费精品| 99视频精品免费观看| 欧美成人高清| 久久蜜桃av| 秋霞影视一区二区三区| 色婷婷亚洲mv天堂mv在影片| 日韩在线视频一区二区三区| 久久精品1区| 综合色就爱涩涩涩综合婷婷| 在线亚洲欧美| 国产视频欧美| 亚洲欧美日韩在线观看a三区| 亚洲精品中文字幕乱码| 免费黄色成人| 久久国产精品99国产| 亚洲精品在线二区| 夜夜嗨一区二区三区| 日韩一区中文| 国产探花一区在线观看| 欧美日一区二区三区在线观看国产免| 日韩va亚洲va欧美va久久| 欧美一区影院| 国产美女精品视频免费播放软件| 天堂va欧美ⅴa亚洲va一国产| 日韩在线麻豆| 欧美天堂一区| 久久影院资源站| 国产自产自拍视频在线观看| 成人欧美一区二区三区的电影| 日本国产精品| 99pao成人国产永久免费视频| 蜜桃视频一区二区三区在线观看| 视频一区视频二区在线观看| 亚洲精品国模| 国产欧美日韩一区二区三区在线| 久久男人av| 99精品视频精品精品视频| 欧美日韩国产综合网| 免费日韩av片| 欧美日韩一区二区三区在线电影| 精品黄色一级片| 亚洲大片在线| 亚洲影院天堂中文av色| 欧美自拍一区| 成人一区不卡| 在线亚洲观看| 久久精品99国产精品| 伊伊综合在线| 国产亚洲精品v| 国产精品主播| 91精品推荐| 亚洲一区欧美| 国产极品一区| 日韩一区二区三区免费播放| 亚洲精品电影| 欧美欧美黄在线二区| 波多野结衣久久精品| 99久久亚洲精品蜜臀| 亚洲一区二区三区高清不卡| 日本天堂一区| 五月激情久久| 亚洲久久视频| 四虎成人av| 日韩一级网站| 国产精品s色| 亚洲高清影视| 国产精品美女在线观看直播| 日韩美女一区二区三区在线观看| 亚洲高清成人| 欧美日韩在线精品一区二区三区激情综合| bbw在线视频| 亚洲精品麻豆| 欧美成人精品三级网站| 天堂久久av| 在线日韩一区| 国产精品午夜av| 国产日韩专区| 国产乱码午夜在线视频| 日韩欧美中文字幕一区二区三区| 国产精品不卡| 日韩中文一区二区| 欧美羞羞视频| 国产免费av国片精品草莓男男 | 日韩国产一二三区| 今天的高清视频免费播放成人| 久久精品99国产精品| 亚洲激情欧美|