要完成properties属性文件某些属性值的加密,和读取属性文件时进行解密
需要4个步骤
- 编写加密解密工具类
- 手动通过加密解密工具类获得加密后的属性值密文,并把密文填写在properties文件中
- 编写PropertyPlaceholderConfigurer的子类,重写convertProperty()方法
- 在spring-dao.xml配置文件中配置PropertyPlaceholderConfigurer类
接下来我们将拿配置数据库的properties文件进行举例(一般我们需要对用户名和密码进行加密)
编写加密解密工具类
在编写工具类前我们需要导入包含Base64这个类的依赖
commons-codec commons-codec 1.14
之所以要使用Base64对加密后的byte数组进行编码,可以参考Base64编码及其作用
编写使用DES加密算法的加密解密工具类
package com.lxc.o2o.util; import java.io.UnsupportedEncodingException; import java.security.Key; import java.security.SecureRandom; import javax.crypto.Cipher; import javax.crypto.KeyGenerator; import org.apache.commons.codec.binary.Base64; /** * DES是一种对称加密算法,所谓对称加密算法即:加密和解密使用相同密钥的算法。 * */ public class DESUtil { // 秘钥对象 private static Key key; // 设置密钥key private static String KEY_STR = "myKey"; // 使用的编码 private static String CHARSETNAME = "UTF-8"; // 设置使用DES算法(我们这里主要使用java的DES算法) private static String ALGORITHM = "DES"; // 初始化秘钥对象key static { try { // 生成DES算法对象 KeyGenerator generator = KeyGenerator.getInstance(ALGORITHM); // 运用SHA1安全策略 SecureRandom secureRandom = SecureRandom.getInstance("SHA1PRNG"); // 设置上密钥种子 secureRandom.setSeed(KEY_STR.getBytes()); // 初始化基于SHA1的算法对象 generator.init(secureRandom); // 生成密钥对象 key = generator.generateKey(); generator = null; } catch (Exception e) { throw new RuntimeException(e); } } /** * 获取加密后的信息 * * @param str * @return */ public static String getEncryptString(String str) { try { // 按UTF8编码 byte[] bytes = str.getBytes(CHARSETNAME); // 获取加密对象 Cipher cipher = Cipher.getInstance(ALGORITHM); // 初始化密码信息,Cipher.ENCRYPT_MODE为加密类型 cipher.init(Cipher.ENCRYPT_MODE, key); // 加密 byte[] doFinal = cipher.doFinal(bytes); // 基于BASE64编码,接收byte[]并转换成String // byte[]to encode好的String并返回,编码成字符串返回 return Base64.encodeBase64String(doFinal); } catch (Exception e) { // TODO: handle exception throw new RuntimeException(e); } } /** * 获取解密之后的信息 * * @param str * @return */ public static String getDecryptString(String str) { try { // 基于BASE64编码,接收byte[]并转换成String // 将字符串decode成byte[],解码操作 byte[] bytes = Base64.decodeBase64(str); // 获取解密对象 Cipher cipher = Cipher.getInstance(ALGORITHM); // 初始化解密信息 cipher.init(Cipher.DECRYPT_MODE, key); // 解密 byte[] doFinal = cipher.doFinal(bytes); // 返回解密之后的信息 return new String(doFinal, CHARSETNAME); } catch (Exception e) { // TODO: handle exception throw new RuntimeException(e); } } public static void main(String[] args) throws UnsupportedEncodingException { System.out.println(getEncryptString("root")); System.out.println(getEncryptString("123456")); } }
获取用户名和密码的秘文
通过上面编写的DESUtil获取用户名和密码的密文,再把密文填写进jdbc.properties文件
jdbc.driver=com.mysql.cj.jdbc.Driver jdbc.url=jdbc:mysql://localhost:3306/myo2o?useSSL=false&serverTimezone=Asia/Shanghai&useUnicode=true&characterEncoding=utf8 jdbc.username=WnplV/ietfQ= jdbc.password=QAHlVoUc49w=
编写PropertyPlaceholderConfigurer的子类
package com.lxc.o2o.util; import org.springframework.beans.factory.config.PropertyPlaceholderConfigurer; /** * 获取解密后的属性值 * * @author L7832 * */ public class EncryptPropertyPlaceholderConfigurer extends PropertyPlaceholderConfigurer { // 需要加密的字段数组(这里整个jdbc属性文件,我们只对username和password加密了) private String[] encryptPropNames = { "jdbc.username", "jdbc.password" }; /** * 对关键的属性进行转换,重写PropertyPlaceholderConfigurer中的convertProperty方法 * 这个函数会对属性文件中所有的属性键值对进行读取 */ @Override protected String convertProperty(String propertyName, String propertyValue) { // 判断属性值是否被加密了 if (isEncryptProp(propertyName)) { // 对已加密的字段进行解密工作 String decryptValue = DESUtil.getDecryptString(propertyValue); return decryptValue; } else { // 如果没被加密,直接返回 return propertyValue; } } /** * 判断该属性是否已加密,主要拿传进来的属性名和上面我们定义的需要加密的字段数组进行比对 * * @param propertyName * @return */ private boolean isEncryptProp(String propertyName) { // 若等于需要加密的field,则进行加密 for (String encryptpropertyName : encryptPropNames) { if (encryptpropertyName.equals(propertyName)) return true; } return false; } }
配置Bean
在spring-dao.xml配置文件中配置我们自己实现的EncryptPropertyPlaceholderConfigurer类
classpath:jdbc.properties
运行时创建了上面的bean后,可以直接通过${属性名}获取解密后的属性值
到这里我们就完成了加密和解密properties文件的所有操作
以上为个人经验,希望能给大家一个参考,也希望大家多多支持IT俱乐部。