位置: IT常识 - 正文

一个注解搞定接口数据脱敏,太强了!(注解注入)

编辑:rootadmin
来源:juejin.cn/post/7110110794188062727 下午惬意时光,突然产品小姐姐走到我面前,打断我短暂的摸鱼time,企图与我进行深入交流,还好我早有防备没有闪,打开瑞star的点单页面,暗示没有一杯coffee解决不了的需求,需求是某些接口返回的信息,涉及到敏感数据的必须进 ...

推荐整理分享一个注解搞定接口数据脱敏,太强了!(注解注入),希望有所帮助,仅作参考,欢迎阅读内容。

文章相关热门搜索词:注解是接口吗,注解注入,注解怎么实现的,注解可以加到接口方法上吗,注解是接口吗,什么是注解 如何定义一个注解,注解是接口吗,怎么定义一个注解,内容如对您有帮助,希望把文章链接给更多的朋友!

来源:juejin.cn/post/7110110794188062727

下午惬意时光,突然产品小姐姐走到我面前,打断我短暂的摸鱼time,企图与我进行深入交流,还好我早有防备没有闪,打开瑞star的点单页面,暗示没有一杯coffee解决不了的需求,需求是某些接口返回的信息,涉及到敏感数据的必须进行脱敏操作,我思考一反,表示某问题,马上安排。

思路

1.要做成可配置多策略的脱敏操作,要不然一个个接口进行脱敏操作,重复的工作量太多,很显然违背了“多写一行算我输”的程序员规范,思来想去,定义数据脱敏注解和数据脱敏逻辑的接口, 在返回类上,对需要进行脱敏的属性加上,并指定对应的脱敏策略操作。

2.接下来我只需要拦截控制器返回的数据,找到带有脱敏注解的属性操作即可,一开始打算用@ControllerAdvice去实现,但发现需要自己去反射类获取注解,当返回对象比较复杂,需要递归去反射,性能一下子就会降低,于是换种思路,我想到平时使用的@JsonFormat,跟我现在的场景很类似,通过自定义注解跟字段解析器,对字段进行自定义解析,tql

代码一个注解搞定接口数据脱敏,太强了!(注解注入)

Spring Boot 基础就不介绍了,推荐下这个实战教程:

https://github.com/javastacks/spring-boot-best-practice

1. 自定义数据注解,并可以配置数据脱敏策略@Target({ElementType.FIELD, ElementType.TYPE})@Retention(RetentionPolicy.RUNTIME)@Documentedpublic @interface DataMasking { DataMaskingFunc maskFunc() default DataMaskingFunc.NO_MASK;}2. 自定义Serializer,参考jackson的StringSerializer,下面的示例只针对String类型进行脱敏public interface DataMaskingOperation { String MASK_CHAR = "*"; String mask(String content, String maskChar);}public enum DataMaskingFunc { /** * 脱敏转换器 */ NO_MASK((str, maskChar) -> { return str; }), ALL_MASK((str, maskChar) -> { if (StringUtils.hasLength(str)) { StringBuilder sb = new StringBuilder(); for (int i = 0; i < str.length(); i++) { sb.append(StringUtils.hasLength(maskChar) ? maskChar : DataMaskingOperation.MASK_CHAR); } return sb.toString(); } else { return str; } }); private final DataMaskingOperation operation; private DataMaskingFunc(DataMaskingOperation operation) { this.operation = operation; } public DataMaskingOperation operation() { return this.operation; }}public final class DataMaskingSerializer extends StdScalarSerializer<Object> { private final DataMaskingOperation operation; public DataMaskingSerializer() { super(String.class, false); this.operation = null; } public DataMaskingSerializer(DataMaskingOperation operation) { super(String.class, false); this.operation = operation; } public boolean isEmpty(SerializerProvider prov, Object value) { String str = (String)value; return str.isEmpty(); } public void serialize(Object value, JsonGenerator gen, SerializerProvider provider) throws IOException { if (Objects.isNull(operation)) { String content = DataMaskingFunc.ALL_MASK.operation().mask((String) value, null); gen.writeString(content); } else { String content = operation.mask((String) value, null); gen.writeString(content); } } public final void serializeWithType(Object value, JsonGenerator gen, SerializerProvider provider, TypeSerializer typeSer) throws IOException { this.serialize(value, gen, provider); } public JsonNode getSchema(SerializerProvider provider, Type typeHint) { return this.createSchemaNode("string", true); } public void acceptJsonFormatVisitor(JsonFormatVisitorWrapper visitor, JavaType typeHint) throws JsonMappingException { this.visitStringFormat(visitor, typeHint); }}3. 自定义AnnotationIntrospector,适配我们自定义注解返回相应的Serializer@Slf4jpublic class DataMaskingAnnotationIntrospector extends NopAnnotationIntrospector { @Override public Object findSerializer(Annotated am) { DataMasking annotation = am.getAnnotation(DataMasking.class); if (annotation != null) { return new DataMaskingSerializer(annotation.maskFunc().operation()); } return null; }}4. 覆盖ObjectMapper@Configuration( proxyBeanMethods = false)public class DataMaskConfiguration { @Configuration( proxyBeanMethods = false ) @ConditionalOnClass({Jackson2ObjectMapperBuilder.class}) static class JacksonObjectMapperConfiguration { JacksonObjectMapperConfiguration() { } @Bean @Primary ObjectMapper jacksonObjectMapper(Jackson2ObjectMapperBuilder builder) { ObjectMapper objectMapper = builder.createXmlMapper(false).build(); AnnotationIntrospector ai = objectMapper.getSerializationConfig().getAnnotationIntrospector(); AnnotationIntrospector newAi = AnnotationIntrospectorPair.pair(ai, new DataMaskingAnnotationIntrospector()); objectMapper.setAnnotationIntrospector(newAi); return objectMapper; } }}5. 返回对象加上注解public class User implements Serializable { /** * 主键ID */ private Long id; /** * 姓名 */ @DataMasking(maskFunc = DataMaskingFunc.ALL_MASK) private String name; /** * 年龄 */ private Integer age; /** * 邮箱 */ @DataMasking(maskFunc = DataMaskingFunc.ALL_MASK) private String email;}

近期热文推荐:

1.1,000+ 道 Java面试题及答案整理(2022最新版)

2.劲爆!Java 协程要来了。。。

3.Spring Boot 2.x 教程,太全了!

4.别再写满屏的爆爆爆炸类了,试试装饰器模式,这才是优雅的方式!!

5.《Java开发手册(嵩山版)》最新发布,速速下载!

觉得不错,别忘了随手点赞+转发哦!

本文链接地址:https://www.jiuchutong.com/zhishi/310645.html 转载请保留说明!

上一篇:分享帝国CMS更改消息提醒的方法(帝国cms如何使用)

下一篇:Uncaught DONException: Failed to execute ‘atob‘ on “window ‘: The string to be decoded is not carrec...

  • 博客怎样进行推广(博客怎样进行推销)

    博客怎样进行推广(博客怎样进行推销)

  • 支付宝账单删除了怎么恢复(支付宝账单删除了怎么看交易记录)

    支付宝账单删除了怎么恢复(支付宝账单删除了怎么看交易记录)

  • 建设银行怎么解除原来设备绑定(建设银行怎么解冻)

    建设银行怎么解除原来设备绑定(建设银行怎么解冻)

  • mac的alt键怎么按(macbook的alt键在哪)

    mac的alt键怎么按(macbook的alt键在哪)

  • vivos1怎么隐藏应用图标(vivo如何隐藏应用软件视频教程)

    vivos1怎么隐藏应用图标(vivo如何隐藏应用软件视频教程)

  • 抖音uid有什么用(抖音uid的作用)

    抖音uid有什么用(抖音uid的作用)

  • 主板电源线怎么拔(主板电源线怎么拆)

    主板电源线怎么拔(主板电源线怎么拆)

  • 复制好的链接在哪里找(复制好的链接在哪里打开)

    复制好的链接在哪里找(复制好的链接在哪里打开)

  • 自动息屏是什么意思

    自动息屏是什么意思

  • 华为mate30pro可以扫描文件吗(华为mate30pro可以用120w充电器吗)

    华为mate30pro可以扫描文件吗(华为mate30pro可以用120w充电器吗)

  • 快手通过私信添加是什么意思(快手私信添加好友)

    快手通过私信添加是什么意思(快手私信添加好友)

  • 抖音怎么开通无限关注(抖音怎么开通无人直播带货)

    抖音怎么开通无限关注(抖音怎么开通无人直播带货)

  • 三星s10 5g版在国内能用吗(三星s10+5g国行)

    三星s10 5g版在国内能用吗(三星s10+5g国行)

  • vivo怎么设置软件权限(vivo手机怎么设置应用锁密码)

    vivo怎么设置软件权限(vivo手机怎么设置应用锁密码)

  • 如何关闭南瓜电影自动续费(如何关闭南瓜电视)

    如何关闭南瓜电影自动续费(如何关闭南瓜电视)

  • 华为mate30怎么打开双击亮屏(华为mate30怎么打开volte)

    华为mate30怎么打开双击亮屏(华为mate30怎么打开volte)

  • vsco安卓需要付费吗(vsco安卓会扣费吗)

    vsco安卓需要付费吗(vsco安卓会扣费吗)

  • 手机wps怎么加粗字体(手机wps怎么加粗横线)

    手机wps怎么加粗字体(手机wps怎么加粗横线)

  • 苹果网络为什么1x(苹果网络为什么这么卡)

    苹果网络为什么1x(苹果网络为什么这么卡)

  • 三星a6060怎么截图(三星a60如何截长图)

    三星a6060怎么截图(三星a60如何截长图)

  • p30pro和mate30pro哪个拍照厉害(p30pro和mate30pro哪个好)

    p30pro和mate30pro哪个拍照厉害(p30pro和mate30pro哪个好)

  • 云黑名单可以解除吗(是云黑名单的一员)

    云黑名单可以解除吗(是云黑名单的一员)

  • vivox27高配和低配的区别(vivox27高配和低配主板外观一样吗)

    vivox27高配和低配的区别(vivox27高配和低配主板外观一样吗)

  • 360路由器手机怎么设置(360路由器手机怎么安装设置)

    360路由器手机怎么设置(360路由器手机怎么安装设置)

  • 苹果iPhone手机显示无SIM卡怎么解决(苹果手机上显示)

    苹果iPhone手机显示无SIM卡怎么解决(苹果手机上显示)

  • 支付给境外个人的租金
  • 购买理财产品是什么意思?
  • 外资研发中心认定标准
  • 私对公转账收税吗
  • 个人所得税红利20%
  • 普票加专票超过30万全交税吗2020
  • 出口资质需要哪些资质
  • 需要冲红的普通发票无法收回怎么办
  • 小规模纳税人开票税率
  • 个人从海外买东西自用要交税吗
  • 退回多打货款分录
  • 收到承兑汇票怎么兑现步骤
  • 当期未开票过收入,以后开票怎么做会计分录
  • 销售业务中的贴息如何开票?
  • 金税盘维护费抵税会计分录
  • 当月计提当月缴纳的增值税还用结转吗
  • 稿费用交个人所得税吗
  • 设置戴尔笔记本锁屏时间
  • windows11怎么显示桌面图标
  • 怎样结转销售成本会计分录
  • 财管如何区分金额大小
  • php获取访问者qq
  • PHP:session_commit()的用法_Session函数
  • win10系统修改密码
  • 客户预付款了尾款怎么办
  • 低值易耗品现在还用吗
  • 公司为员工投保意外险,意外险赔付给谁
  • PHP:imagecolormatch()的用法_GD库图像处理函数
  • 企业成立多久可以注销
  • 微信小程序和web端的交互
  • 旅行社小规模纳税人差额征税
  • 个人转让土地使用权可以开专票吗
  • axios在vue中的使用慕课笔记
  • php正则替换字符串
  • vit详解
  • 基于改进yolov4论文
  • php -a
  • 表单验证用什么方法实现
  • 【历史上的今天】3 月 23 日:网景创始人出生;FORMAC 语言的开发者诞生;PRMan 非商业版发布
  • 计算md5值命令
  • 最新前端技术
  • apt-key is deprecated
  • 充值款如何领取红包
  • 2022年windows11windows10最强VMware Workstation 16 Pro for Windows虚拟机软件最新授权版16.2.3-19376536版本
  • 小规模纳税人怎么核定税种
  • 三十而已 1
  • 财政收入的收缴具体包括
  • 解决Dede5.6联动类型更改排序数字的BUG
  • 如何利用税收经济效应实现税收的宏观调控目标
  • 应收账款收不回来做坏账处理分录
  • 施工企业会计科目
  • 职工教育经费是从工资里扣吗
  • 个人独资企业的
  • 资产处置损益要写明细账吗
  • 进项票可以抵扣企业所得税吗
  • 保险再保业务
  • 暂估成本的账务怎么处理
  • 现金日记账如何填写
  • 什么是企业管理的基础工作
  • 教育经费支出
  • 应收账款周转天数越大说明什么
  • sqlserver的游标
  • xp文件夹选项不见了怎么办
  • windows7如何设置时间格式
  • win7经常断网是什么原因
  • win7微软账户
  • win10升级win1
  • ExtJS扩展 垂直tabLayout实现代码
  • bc1998录制的css视频教程推荐新手看下
  • shell 四则运算
  • shell脚本函数及传参
  • 简单的分享一下
  • c# /n
  • js层级选择器
  • 动态生成xml
  • 安卓的生命周期方法
  • 山西国家税务总局官网
  • 我的电脑为什么没有微软商店
  • 电子发票美元如何查询
  • 甘肃税务厅
  • 免责声明:网站部分图片文字素材来源于网络,如有侵权,请及时告知,我们会第一时间删除,谢谢! 邮箱:opceo@qq.com

    鄂ICP备2023003026号

    网站地图: 企业信息 工商信息 财税知识 网络常识 编程技术

    友情链接: 武汉网站建设