使用SpringBoot+MyBatis實現(xiàn)數(shù)據(jù)庫字段級加密
為什么需要字段級加密?
在當今數(shù)據(jù)驅動的時代,個人信息保護法、網絡安全法等相關法規(guī)相繼出臺,對用戶數(shù)據(jù)安全提出了更高要求。特別是金融、醫(yī)療、教育等行業(yè),用戶的身份信息、聯(lián)系方式、財務數(shù)據(jù)等一旦泄露,后果不堪設想。
傳統(tǒng)做法是直接將數(shù)據(jù)明文存儲在數(shù)據(jù)庫中,這種方式存在巨大安全隱患:
- 數(shù)據(jù)庫泄露風險:一旦數(shù)據(jù)庫被黑客攻破,所有數(shù)據(jù)一覽無余
- 內部人員風險:內部員工可以直接查看敏感數(shù)據(jù)
- 備份泄露風險:數(shù)據(jù)庫備份文件丟失也會導致數(shù)據(jù)泄露
- 合規(guī)風險:不符合GDPR、個人信息保護法等法規(guī)要求
字段級加密正是解決這些問題的有效手段。
技術選型:為什么選擇Spring Boot + MyBatis?
Spring Boot的優(yōu)勢
- 快速開發(fā):約定優(yōu)于配置,極大提升開發(fā)效率
- 生態(tài)完善:豐富的starter組件,開箱即用
- 易于集成:與各類中間件無縫集成
MyBatis的優(yōu)勢
- SQL可控:可以精確控制每一條SQL語句
- 靈活性強:支持復雜查詢和動態(tài)SQL
- 攔截機制:提供了強大的插件機制,便于擴展
核心實現(xiàn)思路
我們的目標是實現(xiàn)一個透明的字段級加密系統(tǒng),整體架構如下:
業(yè)務層 -> MyBatis攔截器 -> 數(shù)據(jù)庫
<- MyBatis攔截器 <-
1. 加密注解設計
首先,我們需要定義一個注解來標記哪些字段需要加密:
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface Encrypt {
}
2. 實體類標記
在實體類中使用注解標記需要加密的字段:
@Data
public class User {
private Long id;
private String username;
@Encrypt
private String password;
@Encrypt
private String email;
@Encrypt
private String phone;
}
3. 加密工具類
實現(xiàn)AES加密算法的工具類:
public class EncryptionUtil {
private static final String ALGORITHM = "AES";
private static final 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) {
throw new 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));
return new String(decryptedBytes, StandardCharsets.UTF_8);
} catch (Exception e) {
throw new RuntimeException("解密失敗", e);
}
}
}
4. MyBatis攔截器實現(xiàn)
這是整個系統(tǒng)的核心部分,通過MyBatis攔截器實現(xiàn)自動加解密:
// 加密攔截器
@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操作進行加密處理
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();
// 對查詢結果進行解密處理
if (result instanceof List) {
List<?> list = (List<?>) result;
for (Object item : list) {
decryptFields(item);
}
} else {
decryptFields(result);
}
return result;
}
}
}
實際應用場景
這套字段級加密系統(tǒng)可以在多個場景中發(fā)揮作用:
1. 用戶信息保護
在用戶注冊時,自動加密用戶的密碼、郵箱、手機號等敏感信息,即使數(shù)據(jù)庫泄露也不會造成用戶隱私泄露。
2. 金融數(shù)據(jù)安全
對用戶的銀行卡號、交易記錄等金融數(shù)據(jù)進行加密存儲,滿足金融行業(yè)的合規(guī)要求。
3. 醫(yī)療數(shù)據(jù)保護
對患者的病歷、診斷結果等醫(yī)療隱私數(shù)據(jù)進行加密,保護患者隱私。
4. 企業(yè)數(shù)據(jù)安全
對企業(yè)內部的商業(yè)機密、客戶資料等重要數(shù)據(jù)進行加密保護。
安全性考慮
雖然字段級加密功能強大,但在生產環(huán)境中使用時必須注意安全性:
- 密鑰管理:不要在代碼中硬編碼密鑰,應使用專業(yè)的密鑰管理系統(tǒng)
- 算法選擇:使用經過驗證的加密算法,如AES-256
- 性能優(yōu)化:合理選擇需要加密的字段,避免對所有字段都進行加密
- 審計日志:記錄所有加密解密操作,便于安全審計
- 定期輪換:定期更換加密密鑰,降低密鑰泄露風險
總結
通過Spring Boot + MyBatis實現(xiàn)數(shù)據(jù)庫字段級加密,我們可以構建一個既安全又透明的數(shù)據(jù)保護系統(tǒng)。這套方案已經在眾多企業(yè)級應用中得到驗證,能夠有效保護用戶敏感數(shù)據(jù)安全,滿足合規(guī)要求。
當然,任何技術都不是銀彈,在享受便利的同時也要注意潛在的風險。希望今天的分享能給大家?guī)硪恍﹩l(fā),讓我們一起探索更多有趣的技術方案!
以上就是使用SpringBoot+MyBatis實現(xiàn)數(shù)據(jù)庫字段級加密的詳細內容,更多關于SpringBoot MyBatis數(shù)據(jù)庫字段級加密的資料請關注腳本之家其它相關文章!
相關文章
Springboot中的@ComponentScan注解使用解析
這篇文章主要介紹了Springboot中的@ComponentScan注解使用解析,@ComponentScan用于類或接口上主要是指定掃描路徑,spring會把指定路徑下帶有指定注解的類注冊到IOC容器中,需要的朋友可以參考下2024-01-01
聊聊Spring循環(huán)依賴三級緩存是否可以減少為二級緩存的情況
這篇文章主要介紹了聊聊Spring循環(huán)依賴三級緩存是否可以減少為二級緩存的情況,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2021-02-02
JVM調優(yōu)OutOfMemoryError異常分析
這篇文章主要為大家介紹了JVM調優(yōu)OutOfMemoryError異常分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2022-11-11
JAVA使用POI獲取Excel的列數(shù)與行數(shù)
Apache POI 是用Java編寫的免費開源的跨平臺的 Java API,Apache POI提供API給Java程式對Microsoft Office格式檔案讀和寫的功能。 下面這篇文章給大家介紹了JAVA使用POI獲取Excel列數(shù)和行數(shù)的方法,有需要的朋友們可以參考借鑒,下面來一起看看吧。2016-12-12
圖解Springboot集成七牛云并實現(xiàn)圖片上傳功能過程
在實際開發(fā)中 ,基本都會有應用到文件上傳的場景,但隨著或多或少的需求問題,之前有在springboot上用過七牛云實現(xiàn)圖片上傳,今天因為某些原因又重新使用了下七牛云因此想總結下七牛云2021-11-11

