当前位置: 首页 > news >正文

对象变更记录objectlog工具

文章目录

    • 前言
    • 演示代码
    • 演示环境
    • 引入项目
      • 项目框架
      • 操作步骤
    • 设计介绍
    • 参考仓库

前言

系统基于mybatis-plus, springboot环境

对于重要的一些数据,我们需要记录一条记录的所有版本变化过程,做到持续追踪,为后续问题追踪提供思路。下面展示预期效果(根据对象field渲染即可):

除了上面这种简单的记录外,还可以记录复合操作,下面举两个例子:

  • 对于批量导入,除了记录对象a本身的变化过程,还维护了父操作id,父操作id我们可以直观了解这个对象a创建的源头,在源头记录中,我们可以获取源头操作所有操作内容(包含了对象a的操作),实现不同对象的联动。
  • 对于单个对象a,若存在关联表字段b对象的修改,会自动将b对象的变更记录和对象a操作记录绑定(当然对象a本身也会产生操作记录)

源代码地址,其中README.md文档中有详细的描述。

演示代码

下面我们通过一段代码演示代码,展示如何自动将枚举字段,主键关联字段,多行文本串,富文本内容等字段自动进行转换和记录(对于其他类型的你可以按照系统规则,自己重写逻辑)。在此期间,你只需要配置两个注解@LogEntity@LogPoint)相关的属性即可实现对象日志的记录,非常方便。

工具采用spring切面和mybatis拦截器相关技术编写了api依赖包,以非侵入方式实现对标记的对象属性进行记录仅需要导入依赖即可,几乎不需要对原系统代码改动

@LogEntity//开启标识publicclassSysUserModelextendsBaseEntity{@LogEntity(alias="角色信息",associationValue=true,serviceImplClass=SysRoleServiceImpl.class,entityFieldName="roleName")//角色表存在 (9830274072323 测试角色) 这一条数据@ApiModelProperty(name="roleId",value="角色表,角色id,逗号隔开")privateStringroleId;@LogEntity(alias="用户昵称")@ApiModelProperty(name="userName",value="用户昵称")privateStringuserName;@LogEntity(alias="备注",attributeTypeEnum=AttributeTypeEnum.TEXT)@ApiModelProperty(name="remark",value="备注")privateStringremark;@LogEntity(alias="富文本内容",attributeTypeEnum=AttributeTypeEnum.RICHTEXT)@ApiModelProperty(name="richText",value="富文本内容")privateStringrichText;@LogEntity(alias="帐号状态",enumValue=true,enumClass=StatusEnum.class)@ApiModelProperty(name="status",value="帐号状态(0正常 1停用)")privateIntegerstatus;}publicenumStatusEnum{NORMAL(0,"正常"),DEACTIVATE(1,"停用"),;//....}
@RestController@RequestMapping(value="/test")publicclassTestController{@ResourceprivateSysUserServiceuserService;@ResourceprivateSysRoleServiceroleService;@ResourceprivateObjectOperationServiceoperationService;@GetMapping(value="/add")@ResponseBodypublicObjecttestAdd(){SysUserModeluserModel=newSysUserModel();userModel.setRoleId("9830274072323");userModel.setUserName("哈哈");userModel.setRemark("这是第一行\n"+"这是第二行");userModel.setStatus(0);userModel.setRichText("<p1>富文本<p1>");userService.add(userModel);}}
@ServicepublicclassSysUserServiceImplextendsServiceImpl<SysUserMapper,SysUserModel>implementsSysUserService{@Override@LogPoint(serviceHandler=SysUserServiceImpl.class,entityHandler=SysUserModel.class,moduleName="sysUser",remark="测试用户模块")publicvoidadd(SysUserModelinsertModel){this.insert(insertModel);}}

结果如下图所示:

{"version":"1.0.0","content":[{"lineNumber":1,"partList":[{"partContent":"这是第一行","partType":"CHANGE_NEW"},{"partContent":"这是第二行","partType":"CHANGE_NEW"}]}]}

可以看到操作很简单,标记两个注解就可以了。

演示环境

  • 导入表 (mysql)

  • 启动nacos(需要发现服务)

  • 启动项目

引入项目

系统现在默认将日志记录api作为依赖集成到某个具体的业务模块中,当然也可以直接将日志做为一个新的服务下面我们按集成到某个具体的业务模块中进行介绍

项目框架

演示系统将核心逻辑拆解为objectlog-apiobjectlog-api-test,其中api是作为依赖引入到你的业务模块中api-test则是演示系统中的业务模块

api依赖中分为core,domain,feign三个包,其中core是核心内容,另外两个根据自身业务可以进行删除和变更

  • core包:核心逻辑的实现
  • domain包:自身业务的基类BaseEntity,自动填充字段逻辑BaseMetaObjectHandler和操作记录查询封装类ObjectOperationDto.
  • feignt包:查询操作记录相关的feign配置

操作步骤

  • 导入表objectlog-api-test模块下的object_attribute.sqlobject_operation.sql导入到业务模块中

  • 改变基类对象,目前主流的mybatis-plug提供了自带curd操作类ServiceImpl,IService,BaseMapper,在@LogPoint中ServiceImpl切换自身业务系统基于ServiceImpl进一步封装类,其次是BaseEntity.class,这个是你自身业务系统的基类。

设计介绍

首先是前文提到的两个注解@LogEntity@LogPoint

  • BASE : 单个的对象记录,不存在子对象的记录
  • COMMENT: 一组对象的记录,不存在父子关系,如批量导入
  • COMPLEX:单个对象的记录,存在子对象的记录

在核心包中有一个handler包,下面提供字段如何字段转换和解析。在包中提供了基础类型(NORMALRICHTEXTTEXT)的类型处理器和值处理器,他们分别实现了AttributeTypeHandlerAttributeValueHandler中的方法,下面我们对其进行一些讲解。

在@LogEntity注解中如果没有指定类型处理器我们发现使用了默认处理器处理,如果你当前系统实现不符合业务逻辑,你可以重写默认处理器中的逻辑即可或者创建一个类继承相关接口,重写逻辑,然后在把重写的类赋予注解中的值即可

源代码地址,感兴趣的朋友gitee上提issue单,后面持续补充。

参考仓库

原仓库只是简单的实现,入侵业务严重,不过对于本系统中RICHTEXT,TEXT类型的字段提供了思路 参考仓库

http://www.jsqmd.com/news/799369/

相关文章:

  • ARM Firmware Suite与Integrator开发板嵌入式开发指南
  • Super Plan Mode:AI编程助手的“计划优先”协作协议
  • 小白转行谷歌SEO:发展前景与薪资解析
  • Halcon实战:傅里叶变换滤波器的参数调优与视觉应用
  • Linux桌面效率神器:X11事件驱动实现鼠标自动跟随焦点窗口
  • STM32H7实战:用FMC+DMA双缓冲搞定AD7606,8通道数据采集稳如老狗
  • 从零到一掌握AKS实战:集群部署、应用容器化与CI/CD全流程详解
  • Sycamore框架实战:复杂文档RAG系统构建与优化指南
  • 基于计算机视觉的游戏AI开发:Agent of Empires框架实战解析
  • 技能使用分析工具:从日志复盘到数据驱动的效率优化
  • STM32F103实战:IIC协议驱动SHT31实现高精度环境监测
  • 高盛:AI是否泡沫,要看接下来5年的7.6万亿美元基建
  • Perplexity真能取代Google?2024年实测对比:响应深度、事实准确率、代码理解力三大维度揭密
  • ARM GIC-500中断控制器原理与勘误实战解析
  • Arm编译器嵌入式开发:线程安全与浮点运算实战
  • 在 Linux 下怎么查看谁在使用 80 端口?
  • 详解 Deepsec:Vercel 开源 AI 代码安全防护工具的技术架构与实现原理
  • 【计算机毕业设计】基于Springboot的纺织品企业财务管理系统设计与实现+LW
  • 【WPF】Blend实战:从零构建流畅UI动画
  • qt5.14.2连mysql8.0
  • ARM926EJ-S指令缓存架构与调试技术详解
  • C# 绘制直线 圆形 矩形(工业上位机)
  • 【数学建模】雾霾问题的建模和仿真分析的MATLAB代码
  • 文献阅读 260511-Wildfire damages and the cost-effective role of forest fuel treatments
  • 基于MCP协议实现AI助手个性化:Terminal Buddies项目实战解析
  • 【计算机毕业设计】基于Springboot的医院后台管理系统设计与实现+LW
  • 小白也能上手!OpenClaw 2.6.4 Windows 一键部署本地 AI 智能体
  • NCCL watchdog timeout 先别只会加 timeout:PyTorch 新出的 Flight Recorder,真正值钱的是能把第一处 collective 分歧揪出来
  • 时序数据库查询新思路:用InfluxDB的SELECT、LIMIT、OFFSET玩转IoT设备历史数据分页
  • 工厂6S搞了没效果?精益生产6S红牌作战实操,30天打造标杆车间!