位置: 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...

  • 拼多多在日历上的日程怎么取消(拼多多那个日历怎么弄掉)

    拼多多在日历上的日程怎么取消(拼多多那个日历怎么弄掉)

  • 魅族18在屏幕新增了哪些功能(魅族18手机屏)

    魅族18在屏幕新增了哪些功能(魅族18手机屏)

  • 小米盒子双清重启无效(小米盒子双清重启无效刷机教程)

    小米盒子双清重启无效(小米盒子双清重启无效刷机教程)

  • 荣耀手环4对比小米手环4(荣耀手环对比图)

    荣耀手环4对比小米手环4(荣耀手环对比图)

  • 苹果11实况是什么意思(iphone 11的实况照片是什么)

    苹果11实况是什么意思(iphone 11的实况照片是什么)

  • 淘宝给卖家发消息叹号(淘宝给卖家发消息发不过去)

    淘宝给卖家发消息叹号(淘宝给卖家发消息发不过去)

  • 怎样删除oppo手机使用所有应用记录时间(怎样删除OPPO手机隐私空间的密码)

    怎样删除oppo手机使用所有应用记录时间(怎样删除OPPO手机隐私空间的密码)

  • 苹果手机与电脑连接的软件(苹果手机与电脑连接不上)

    苹果手机与电脑连接的软件(苹果手机与电脑连接不上)

  • 探探没上线会更新位置距离吗(探探上的人不上线怎么办)

    探探没上线会更新位置距离吗(探探上的人不上线怎么办)

  • 苹果x是什么系统(苹果x属于苹果几系列)

    苹果x是什么系统(苹果x属于苹果几系列)

  • oppor17和r17pro区别(oppor17和r17pro区别图片)

    oppor17和r17pro区别(oppor17和r17pro区别图片)

  • 为什么抖音上传的照片很模糊(为什么抖音上传的视频很卡)

    为什么抖音上传的照片很模糊(为什么抖音上传的视频很卡)

  • 手机卡机关不了机怎么办(手机卡机关不了机滑动也滑不了)

    手机卡机关不了机怎么办(手机卡机关不了机滑动也滑不了)

  • 手机qq聊天记录如何恢复(手机qq聊天记录删除了怎么恢复)

    手机qq聊天记录如何恢复(手机qq聊天记录删除了怎么恢复)

  • airpods只能连苹果吗(airpods只能连苹果耳机吗)

    airpods只能连苹果吗(airpods只能连苹果耳机吗)

  • 铁塔集团5G授权给哪些公司(铁塔集团5g授权给哪些公司)

    铁塔集团5G授权给哪些公司(铁塔集团5g授权给哪些公司)

  • 十进制小数转十六进制(十进制小数转十二进制)

    十进制小数转十六进制(十进制小数转十二进制)

  • ps修照片基本步骤人像(ps里修照片)

    ps修照片基本步骤人像(ps里修照片)

  • Win11系统更新卡在100%怎么办 win11更新卡住的解决方法(win11系统更新卡在25%)

    Win11系统更新卡在100%怎么办 win11更新卡住的解决方法(win11系统更新卡在25%)

  • 人工智能导论(第四版)王万良编著课后习题答案(人工智能导论报告)

    人工智能导论(第四版)王万良编著课后习题答案(人工智能导论报告)

  • 手相算命图解大全(手相算命图解大全女)

    手相算命图解大全(手相算命图解大全女)

  • vue/react项目刷新页面出现404的原因以及解决办法(react 刷新)

    vue/react项目刷新页面出现404的原因以及解决办法(react 刷新)

  • 资产处置出售
  • 如何确定关联方及关联关系
  • 消费税的会计分局
  • 如何查询增值税申报表
  • 已认证的专票可以取消认证吗
  • 太阳能发电税收优惠政策
  • 哪些发票可以抵扣税
  • 低值易耗品包括哪些东西和产品
  • 工程款转入冻结账户能拿回
  • 会计科目已受控于应收应付系统
  • 一个月作废发票多会有影响吗?
  • 期末库存农产品进项转出如何算
  • 物业公司收取电损费合法吗
  • 质押费用计入什么科目
  • 商业汇票的样本
  • 押金放到哪个会计科目
  • 实收资本转出计入什么科目
  • 应付职工薪酬月末结转到哪里
  • 公司进货可以计提折旧吗
  • 企业公车私用如何处理
  • 已出账但未认证的抵扣联怎么办?
  • 消费税暂行条例实施细则
  • 福利费用不用计提
  • 外购软件可以加计扣除吗
  • 无票收入已报税,后来开票账务处理
  • 事业基金弥补收支差额
  • 坏账准备的转回对资产的影响
  • 营改增建筑工程怎么计算举例
  • 研发支出和研发费用是一个吗
  • 利润的计算公式excel
  • 电脑非法关机后开机进不了系统
  • 原材料赔偿会计分录
  • PHP:pg_client_encoding()的用法_PostgreSQL函数
  • 期货公司向客户收取的保证金属于谁所有
  • 预付款不退如何投诉
  • 如何理解什么是社会制度
  • 作废的发票对方钱能打出来吗
  • 采购过程中发生材料毁损,由保险公司赔偿的部分
  • 建安企业账务处理 工程施工
  • mysql如何避免锁表
  • python 命令行参数解析
  • 工业总产值和营业收入区别
  • 小规模纳税人广告税率是多少
  • 代理付银行手续费合法吗
  • 红字发票申请单怎么开
  • 利润是非限定净现值吗
  • 发票含税和不含税的区别
  • 存货计提减值准备对所得税的影响
  • 个别计价法下,把每一种存货的实际成本
  • 关税完税价格计算增值税
  • 股东权益合计等于净资产吗
  • 苗圃的账务处理
  • 咨询费收入成本怎么算
  • 佣金和其他费用
  • 货款去零头分录
  • 营改增对财务人员的影响
  • 资产减值损失是负的意味着什么
  • 合伙企业与公司相比,有什么优势呢?
  • 出纳可以做库管吗
  • mysql -ne
  • sql server发布
  • sql2008开启远程连接
  • 关于国际学校
  • nvidia发布了G6X
  • linux 查看so
  • 演示模式怎么设置
  • realshed.exe - realshed是什么进程 有什么用
  • win7玩穿越火线电脑应该怎么设置
  • win7进运行
  • w10消费者版本和专业版区别
  • glib库
  • 在机上创建一个文件夹
  • python模拟登录爬取数据
  • opengl实现光线追踪
  • Node.js中的事件循环是什么意思
  • python如何处理文本
  • jQuery ajax全局函数处理session过期后的ajax跳转问题
  • 厦门象屿正式员工招聘
  • 税务打虚打骗
  • 三免三减半政策文件
  • 免责声明:网站部分图片文字素材来源于网络,如有侵权,请及时告知,我们会第一时间删除,谢谢! 邮箱:opceo@qq.com

    鄂ICP备2023003026号

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

    友情链接: 武汉网站建设