前言

在实际开发的过程中,我们可能会遇到一些敏感数据,但是用户又不希望这些敏感数据全部展示出来,所以这个时候就需要对数据进行脱敏处理


解决思路:新建一个注解了标签,用在实体类字段中,实现再查询数据时将数据替换成处理好的数据

一、处理步骤

1. 新建脱敏的枚举类

  1. package com.fwy.common.config.dataRule;
  2. import lombok.Getter;
  3. /**
  4. * @description:隐私数据类型枚举
  5. * @author: xiaYZ
  6. * @createDate: 2022/6/21
  7. */
  8. @Getter
  9. public enum PrivacyTypeEnum {
  10. /** 自定义(此项需设置脱敏的范围)*/
  11. CUSTOMER,
  12. /** 姓名 */
  13. NAME,
  14. /** 身份证号 */
  15. ID_CARD,
  16. /** 手机号 */
  17. PHONE,
  18. /** 邮箱 */
  19. EMAIL,
  20. }

2.新建脱敏操作的工具类

  1. package com.fwy.common.config.dataRule;
  2. /**
  3. * @description:
  4. * @author: xiaYZ
  5. * @createDate: 2022/6/21
  6. */
  7. public class PrivacyUtil {
  8. /**
  9. * 隐藏手机号中间四位
  10. */
  11. public static String hidePhone(String phone) {
  12. return phone.replaceAll("(\\d{3})\\d{4}(\\d{4})", "$1****$2");
  13. }
  14. /**
  15. * 隐藏邮箱
  16. */
  17. public static String hideEmail(String email) {
  18. return email.replaceAll("(\\w?)(\\w+)(\\w)(@\\w+\\.[a-z]+(\\.[a-z]+)?)", "$1****$3$4");
  19. }
  20. /**
  21. * 隐藏身份证
  22. */
  23. public static String hideIDCard(String idCard) {
  24. return idCard.replaceAll("(\\d{4})\\d{10}(\\w{4})", "$1*****$2");
  25. }
  26. /**
  27. * 【中文姓名】只显示第一个汉字,其他隐藏为星号,比如:任**
  28. */
  29. public static String hideChineseName(String chineseName) {
  30. if (chineseName == null) {
  31. return null;
  32. }
  33. return desValue(chineseName, 1, 0, "*");
  34. }
  35. // /**
  36. // * 【身份证号】显示前4位, 后2位
  37. // */
  38. // public static String hideIdCard(String idCard) {
  39. // return desValue(idCard, 4, 2, "*");
  40. // }
  41. // /**
  42. // * 【手机号码】前三位,后四位,其他隐藏。
  43. // */
  44. // public static String hidePhone(String phone) {
  45. // return desValue(phone, 3, 4, "*");
  46. // }
  47. /**
  48. * 对字符串进行脱敏操作
  49. * @param origin 原始字符串
  50. * @param prefixNoMaskLen 左侧需要保留几位明文字段
  51. * @param suffixNoMaskLen 右侧需要保留几位明文字段
  52. * @param maskStr 用于遮罩的字符串, 如'*'
  53. * @return 脱敏后结果
  54. */
  55. public static String desValue(String origin, int prefixNoMaskLen, int suffixNoMaskLen, String maskStr) {
  56. if (origin == null) {
  57. return null;
  58. }
  59. StringBuilder sb = new StringBuilder();
  60. for (int i = 0, n = origin.length(); i < n; i++) {
  61. if (i < prefixNoMaskLen) {
  62. sb.append(origin.charAt(i));
  63. continue;
  64. }
  65. if (i > (n - suffixNoMaskLen - 1)) {
  66. sb.append(origin.charAt(i));
  67. continue;
  68. }
  69. sb.append(maskStr);
  70. }
  71. return sb.toString();
  72. }
  73. }

3. 申明注解类

  1. package com.fwy.common.config.dataRule;
  2. import com.fasterxml.jackson.annotation.JacksonAnnotationsInside;
  3. import com.fasterxml.jackson.databind.annotation.JsonSerialize;
  4. import java.lang.annotation.ElementType;
  5. import java.lang.annotation.Retention;
  6. import java.lang.annotation.RetentionPolicy;
  7. import java.lang.annotation.Target;
  8. /**
  9. * @description: 自定义数据脱敏注解
  10. * @author: xiaYZ
  11. * @createDate: 2022/6/21
  12. */
  13. @Target(ElementType.FIELD) // 作用在字段上
  14. @Retention(RetentionPolicy.RUNTIME) // class文件中保留,运行时也保留,能通过反射读取到
  15. @JacksonAnnotationsInside // 表示自定义自己的注解PrivacyEncrypt
  16. @JsonSerialize(using = PrivacySerializer.class) // 该注解使用序列化的方式
  17. public @interface PrivacyEncrypt {
  18. /**
  19. * 脱敏数据类型(没给默认值,所以使用时必须指定type)
  20. */
  21. PrivacyTypeEnum type();
  22. /**
  23. * 前置不需要打码的长度
  24. */
  25. int prefixNoMaskLen() default 1;
  26. /**
  27. * 后置不需要打码的长度
  28. */
  29. int suffixNoMaskLen() default 1;
  30. /**
  31. * 用什么打码
  32. */
  33. String symbol() default "*";
  34. }

4.注解的AOP操作

  1. package com.fwy.common.config.dataRule;
  2. import com.fasterxml.jackson.core.JsonGenerator;
  3. import com.fasterxml.jackson.databind.BeanProperty;
  4. import com.fasterxml.jackson.databind.JsonMappingException;
  5. import com.fasterxml.jackson.databind.JsonSerializer;
  6. import com.fasterxml.jackson.databind.SerializerProvider;
  7. import com.fasterxml.jackson.databind.ser.ContextualSerializer;
  8. import lombok.AllArgsConstructor;
  9. import lombok.NoArgsConstructor;
  10. import org.apache.commons.lang3.StringUtils;
  11. import java.io.IOException;
  12. import java.util.Objects;
  13. /**
  14. * @description:
  15. * @author: xiaYZ
  16. * @createDate: 2022/6/21
  17. */
  18. @NoArgsConstructor
  19. @AllArgsConstructor
  20. public class PrivacySerializer extends JsonSerializer<String> implements ContextualSerializer {
  21. // 脱敏类型
  22. private PrivacyTypeEnum privacyTypeEnum;
  23. // 前几位不脱敏
  24. private Integer prefixNoMaskLen;
  25. // 最后几位不脱敏
  26. private Integer suffixNoMaskLen;
  27. // 用什么打码
  28. private String symbol;
  29. @Override
  30. public void serialize(final String origin, final JsonGenerator jsonGenerator,
  31. final SerializerProvider serializerProvider) throws IOException {
  32. if (StringUtils.isNotBlank(origin) && null != privacyTypeEnum) {
  33. switch (privacyTypeEnum) {
  34. case CUSTOMER:
  35. jsonGenerator.writeString(PrivacyUtil.desValue(origin, prefixNoMaskLen, suffixNoMaskLen, symbol));
  36. break;
  37. case NAME:
  38. jsonGenerator.writeString(PrivacyUtil.hideChineseName(origin));
  39. break;
  40. case ID_CARD:
  41. jsonGenerator.writeString(PrivacyUtil.hideIDCard(origin));
  42. break;
  43. case PHONE:
  44. jsonGenerator.writeString(PrivacyUtil.hidePhone(origin));
  45. break;
  46. case EMAIL:
  47. jsonGenerator.writeString(PrivacyUtil.hideEmail(origin));
  48. break;
  49. default:
  50. throw new IllegalArgumentException("unknown privacy type enum " + privacyTypeEnum);
  51. }
  52. }
  53. }
  54. @Override
  55. public JsonSerializer<?> createContextual(final SerializerProvider serializerProvider,
  56. final BeanProperty beanProperty) throws JsonMappingException {
  57. if (beanProperty != null) {
  58. if (Objects.equals(beanProperty.getType().getRawClass(), String.class)) {
  59. PrivacyEncrypt privacyEncrypt = beanProperty.getAnnotation(PrivacyEncrypt.class);
  60. if (privacyEncrypt == null) {
  61. privacyEncrypt = beanProperty.getContextAnnotation(PrivacyEncrypt.class);
  62. }
  63. if (privacyEncrypt != null) {
  64. return new PrivacySerializer(privacyEncrypt.type(), privacyEncrypt.prefixNoMaskLen(),
  65. privacyEncrypt.suffixNoMaskLen(), privacyEncrypt.symbol());
  66. }
  67. }
  68. return serializerProvider.findValueSerializer(beanProperty.getType(), beanProperty);
  69. }
  70. return serializerProvider.findNullValueSerializer(null);
  71. }
  72. }

5.实际使用

springBoot中如何给敏感数据脱敏 - 图1
springBoot中如何给敏感数据脱敏 - 图2

6.可能遇到问题

  1. 如果在保存数据的时候使用此注解的类会把处理好的数据保存到数据库从而导致数据不准确
  2. 有时候查询的数据需要部署脱敏的数据而是原数据
    解决:可以多声明一个类的字段,如idCardNumber2代表脱敏数据,而idCardNumber代表原数据即可

总结

对数据进行脱敏操作,总结就是使用AOP对数据进行拦截操作,在给对象赋值时进行数据操作