SpringBoot結(jié)合MyBatis實(shí)現(xiàn)數(shù)據(jù)庫字段加密
今天分享一篇基于SpringBoot+MyBatis來實(shí)現(xiàn)數(shù)據(jù)庫字段加密的操作,喜歡的朋友可以借鑒
大致的實(shí)現(xiàn)流程
業(yè)務(wù)層-->系統(tǒng)攔截器-->數(shù)據(jù)庫-->系統(tǒng)攔截器-->返回結(jié)果
加密注解設(shè)計(jì)
把需要加密的字段通過我們自定義的加密注解進(jìn)行標(biāo)識(shí),所以我們需要先自定義一段加密注解的代碼
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface Encrypt {
}實(shí)體類
在實(shí)體類上使用注解標(biāo)記需要加密字段
@Data
public class User {
private Long id;
private String username;
@Encrypt
private String password;
@Encrypt
private String email;
@Encrypt
private String phone;
}加密工具類
基于AES加密算法實(shí)現(xiàn)對字段名的加密,大家可以選擇其他的加密算法
public class EncryptionUtil {
privatestaticfinal String ALGORITHM = "AES";
privatestaticfinal String TRANSFORMATION = "AES/ECB/PKCS5Padding";
// AES加密
public static String encrypt(String plainText, String key) {
try {
SecretKeySpec secretKey = new SecretKeySpec(key.getBytes(StandardCharsets.UTF_8), ALGORITHM);
Cipher cipher = Cipher.getInstance(TRANSFORMATION);
cipher.init(Cipher.ENCRYPT_MODE, secretKey);
byte[] encryptedBytes = cipher.doFinal(plainText.getBytes(StandardCharsets.UTF_8));
return Base64.getEncoder().encodeToString(encryptedBytes);
} catch (Exception e) {
thrownew RuntimeException("加密失敗", e);
}
}
// AES解密
public static String decrypt(String cipherText, String key) {
try {
SecretKeySpec secretKey = new SecretKeySpec(key.getBytes(StandardCharsets.UTF_8), ALGORITHM);
Cipher cipher = Cipher.getInstance(TRANSFORMATION);
cipher.init(Cipher.DECRYPT_MODE, secretKey);
byte[] decryptedBytes = cipher.doFinal(Base64.getDecoder().decode(cipherText));
returnnew String(decryptedBytes, StandardCharsets.UTF_8);
} catch (Exception e) {
thrownew RuntimeException("解密失敗", e);
}
}
}系統(tǒng)攔截器設(shè)計(jì)
通過攔截實(shí)現(xiàn)自動(dòng)加密和自動(dòng)解密
// 加密攔截器
@Intercepts({
@Signature(type = Executor.class, method = "update", args = {MappedStatement.class, Object.class})
})
@Component
public class FieldEncryptionInterceptor implements Interceptor {
@Value("${encryption.key:mySecretKey12345}")
private String encryptionKey;
@Override
public Object intercept(Invocation invocation) throws Throwable {
Object[] args = invocation.getArgs();
MappedStatement mappedStatement = (MappedStatement) args[0];
Object parameter = args[1];
// 獲取SQL命令類型
String sqlCommandType = mappedStatement.getSqlCommandType().toString();
// 對INSERT和UPDATE操作進(jìn)行加密處理
if ("INSERT".equals(sqlCommandType) || "UPDATE".equals(sqlCommandType)) {
encryptFields(parameter);
}
return invocation.proceed();
}
// 解密攔截器
@Intercepts({
@Signature(type = ResultSetHandler.class, method = "handleResultSets", args = {Statement.class})
})
@Component
public class FieldDecryptionInterceptor implements Interceptor {
@Value("${encryption.key:mySecretKey12345}")
private String encryptionKey;
@Override
public Object intercept(Invocation invocation) throws Throwable {
// 執(zhí)行原始方法
Object result = invocation.proceed();
// 對查詢結(jié)果進(jìn)行解密處理
if (result instanceof List) {
List<?> list = (List<?>) result;
for (Object item : list) {
decryptFields(item);
}
} else {
decryptFields(result);
}
return result;
}
}
}測試場景
- 用戶信息保護(hù):在用戶注冊時(shí),自動(dòng)加密用戶的密碼、郵箱、手機(jī)號(hào)等敏感信息,即使數(shù)據(jù)庫泄露也不會(huì)造成用戶隱私泄露
- 金融數(shù)據(jù)保護(hù):對用戶的銀行卡號(hào)、交易記錄等金融數(shù)據(jù)進(jìn)行加密存儲(chǔ),滿足金融行業(yè)的合規(guī)要求
- 醫(yī)療醫(yī)保數(shù)據(jù)保護(hù):對患者的病歷、診斷結(jié)果等醫(yī)療隱私數(shù)據(jù)進(jìn)行加密,保護(hù)患者隱私
- 企業(yè)數(shù)據(jù)保護(hù):對企業(yè)內(nèi)部的商業(yè)機(jī)密、客戶資料等重要數(shù)據(jù)進(jìn)行加密保護(hù)
注意事項(xiàng)
- 雖然字段級(jí)加密功能強(qiáng)大,但在生產(chǎn)環(huán)境中使用時(shí)必須注意安全性:
- 密鑰管理:不要在代碼中硬編碼密鑰,應(yīng)使用專業(yè)的密鑰管理系統(tǒng)
- 算法選擇:使用經(jīng)過驗(yàn)證的加密算法,如AES-256
- 性能優(yōu)化:合理選擇需要加密的字段,避免對所有字段都進(jìn)行加密
- 審計(jì)日志:記錄所有加密解密操作,便于安全審計(jì)
- 定期輪換:定期更換加密密鑰,降低密鑰泄露風(fēng)險(xiǎn)
到此這篇關(guān)于SpringBoot結(jié)合MyBatis實(shí)現(xiàn)數(shù)據(jù)庫字段加密的文章就介紹到這了,更多相關(guān)SpringBoot MyBatis數(shù)據(jù)庫字段加密內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java發(fā)送form-data請求的實(shí)例代碼
在Java中發(fā)送form-data請求,可以使用Apache?HttpClient或OkHttp這樣的HTTP客戶端庫來發(fā)送請求,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友參考下吧2023-10-10
java中switch case語句需要加入break的原因解析
這篇文章主要介紹了java中switch case語句需要加入break的原因解析的相關(guān)資料,需要的朋友可以參考下2017-07-07
MyBatis CodeHelperPro激活方法詳細(xì)教程
MyBatisCodeHelper-Pro是IDEA下的一個(gè)插件,功能類似mybatis plugin,今天小編給大家分享MyBatis CodeHelperPro激活方法,需要的朋友跟隨小編一起看看吧2021-07-07
在Java中實(shí)現(xiàn)線程安全的單例模式的常見方式
單例模式是一種常用的軟件設(shè)計(jì)模式,它確保一個(gè)類只有一個(gè)實(shí)例,并提供一個(gè)全局訪問點(diǎn),在多線程環(huán)境下,確保單例模式的線程安全性是非常重要的,因?yàn)槎鄠€(gè)線程可能會(huì)同時(shí)嘗試創(chuàng)建實(shí)例,導(dǎo)致實(shí)例不唯一的問題,本文介紹了在Java中實(shí)現(xiàn)線程安全的單例模式有幾種常見的方式2024-09-09
詳談@Cacheable不起作用的原因:bean未序列化問題
這篇文章主要介紹了@Cacheable不起作用的原因:bean未序列化問題,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-01-01
java學(xué)習(xí)之利用TCP實(shí)現(xiàn)的簡單聊天示例代碼
這篇文章主要給大家介紹了關(guān)于java學(xué)習(xí)筆記之利用TCP實(shí)現(xiàn)的簡單聊天的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧。2017-12-12
SpringBoot接口訪問頻率限制的實(shí)現(xiàn)方式
接口訪問頻率限制是通過在一定時(shí)間內(nèi)限制用戶對接口的訪問次數(shù)來實(shí)現(xiàn)的,在Spring Boot中,我們可以通過多種方式來實(shí)現(xiàn)接口的限流,如使用過濾器、攔截器或者借助第三方庫,本文給大家講解的非常詳細(xì),需要的朋友可以參考下2024-07-07
在Eclipse安裝Spring boot插件的步驟(圖文)
這篇文章主要介紹了在Eclipse安裝Spring boot插件的步驟,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-08-08

