• SpringBoot项目如何优雅的实现操作日志记录


    前言

    大家好,我是希留。

    在实际开发当中,对于某些关键业务,我们通常需要记录该操作的内容,一个操作调一次记录方法,每次还得去收集参数等等,会造成大量代码重复。 我们希望代码中只有业务相关的操作,在项目中使用注解来完成此项功能。

    通常就是使用Spring中的AOP特性来实现的,那么在SpringBoot项目当中应该如何来实现呢?


    一、AOP是什么?

    AOP(Aspect-Oriented Programming:⾯向切⾯编程),说起AOP,几乎学过Spring框架的人都知道,它是Spring的三大核心思想之一(IOC:控制反转,DI:依赖注入,AOP:面向切面编程)。能够将那些与业务⽆关,却为业务模块所共同调⽤的逻辑或责任(例如事务处理、⽇志管理、权限控制等)封装起来,便于减少系统的重复代码,降低模块间的耦合度,并有利于未来的可拓展性和可维护性。

    二、AOP做了什么?

    简单说来,AOP主要做三件事:

    • 1、在哪里切入,也就是日志记录等非业务代码在哪些业务代码中执行。
    • 2、在什么时候切入,是在业务代码执行前还是后。
    • 3、切入后做什么事情,比如权限校验,日志记录等。

    可以用一张图来理解:

    图上的一个核心术语的说明:

    • Pointcut切点,决定在何处切入业务代码中(即织入切面)。切点分为execution方式和annotation方式。execution方式:可以用路径表达式指定哪些类织入切面,annotation方式:可以指定被哪些注解修饰的代码织入切面。
    • Advice处理,包括处理时机和处理内容。处理内容就是要做什么事,比如校验权限和记录日志。处理时机就是在什么时机执行处理内容,分为前置处理(即业务代码执行前)、后置处理(业务代码执行后)等。
    • Aspect切面,即Pointcut和Advice。
    • Joint point连接点,是程序执行的一个点。例如,一个方法的执行或者一个异常的处理。在 Spring AOP 中,一个连接点总是代表一个方法执行。
    • Weaving织入,就是通过动态代理,在目标对象方法中执行处理内容的过程。

    三、实现步骤

    (1)自定义一个注解@Log (2)创建一个切面类,切点设置为拦截标注@Log的方法,截取传参,进行日志记录 (3)将@Log标注在接口上

    具体的实现步骤如下:

    1. 添加AOP依赖

    代码如下(示例):

    1. <dependency>
    2. <groupId>org.springframework.boot</groupId>
    3. <artifactId>spring-boot-starter-aop</artifactId>
    4. </dependency>
    5. 复制代码

    2. 自定义一个日志注解

    日志一般使用的是注解类型的切点表达式,我们先创建一个日志注解,当spring容器扫描到有此注解的方法就会进行增强。代码如下(示例):

    1. @Target({ ElementType.PARAMETER, ElementType.METHOD }) // 注解放置的目标位置,PARAMETER: 可用在参数上 METHOD:可用在方法级别上
    2. @Retention(RetentionPolicy.RUNTIME) // 指明修饰的注解的生存周期 RUNTIME:运行级别保留
    3. @Documented
    4. public @interface Log {
    5. /**
    6. * 模块
    7. */
    8. String title() default "";
    9. /**
    10. * 功能
    11. */
    12. public BusinessType businessType() default BusinessType.OTHER;
    13. /**
    14. * 是否保存请求的参数
    15. */
    16. public boolean isSaveRequestData() default true;
    17. /**
    18. * 是否保存响应的参数
    19. */
    20. public boolean isSaveResponseData() default true;
    21. }
    22. 复制代码

    3. 切面声明

    申明一个切面类,并交给Spring容器管理。代码如下(示例):

    1. @Aspect
    2. @Component
    3. @Slf4j
    4. public class LogAspect {
    5. @Autowired
    6. private IXlOperLogService operLogService;
    7. /**
    8. * 处理完请求后执行
    9. * @param joinPoint 切点
    10. */
    11. @AfterReturning(pointcut = "@annotation(controllerLog)", returning = "jsonResult")
    12. public void doAfterReturnibng(JoinPoint joinPoint, Log controllerLog, Object jsonResult) {
    13. handleLog(joinPoint, controllerLog, null, jsonResult);
    14. }
    15. protected void handleLog(final JoinPoint joinPoint, Log controllerLog, final Exception e, Object jsonResult) {
    16. try {
    17. // 获取当前的用户
    18. JwtUser loginUser = SecurityUtils.getLoginUser();
    19. // 日志记录
    20. XlOperLog operLog = new XlOperLog();
    21. operLog.setStatus(0);
    22. // 请求的IP地址
    23. String iP = ServletUtil.getClientIP(ServletUtils.getRequest());
    24. if ("0:0:0:0:0:0:0:1".equals(iP)) {
    25. iP = "127.0.0.1";
    26. }
    27. operLog.setOperIp(iP);
    28. operLog.setOperUrl(ServletUtils.getRequest().getRequestURI());
    29. if (loginUser != null) {
    30. operLog.setOperName(loginUser.getUsername());
    31. }
    32. if (e != null) {
    33. operLog.setStatus(1);
    34. operLog.setErrorMsg(StringUtils.substring(e.getMessage(), 0, 2000));
    35. }
    36. // 设置方法名称
    37. String className = joinPoint.getTarget().getClass().getName();
    38. String methodName = joinPoint.getSignature().getName();
    39. operLog.setMethod(className + "." + methodName + "()");
    40. operLog.setRequestMethod(ServletUtils.getRequest().getMethod());
    41. operLog.setOperTime(new Date());
    42. // 处理设置注解上的参数
    43. getControllerMethodDescription(joinPoint, controllerLog, operLog, jsonResult);
    44. // 保存数据库
    45. operLogService.save(operLog);
    46. } catch (Exception exp) {
    47. log.error("异常信息:{}", exp.getMessage());
    48. exp.printStackTrace();
    49. }
    50. }
    51. /**
    52. * 获取注解中对方法的描述信息 用于Controller层注解
    53. * @param log 日志
    54. * @param operLog 操作日志
    55. * @throws Exception
    56. */
    57. public void getControllerMethodDescription(JoinPoint joinPoint, Log log, XlOperLog operLog, Object jsonResult) throws Exception {
    58. // 设置操作业务类型
    59. operLog.setBusinessType(log.businessType().ordinal());
    60. // 设置标题
    61. operLog.setTitle(log.title());
    62. // 是否需要保存request,参数和值
    63. if (log.isSaveRequestData()) {
    64. // 设置参数的信息
    65. setRequestValue(joinPoint, operLog);
    66. }
    67. // 是否需要保存response,参数和值
    68. if (log.isSaveResponseData() && StringUtils.isNotNull(jsonResult)) {
    69. operLog.setJsonResult(StringUtils.substring(JSON.toJSONString(jsonResult), 0, 2000));
    70. }
    71. }
    72. /**
    73. * 获取请求的参数,放到log中
    74. * @param operLog 操作日志
    75. * @throws Exception 异常
    76. */
    77. private void setRequestValue(JoinPoint joinPoint, XlOperLog operLog) throws Exception {
    78. String requsetMethod = operLog.getRequestMethod();
    79. if (HttpMethod.PUT.name().equals(requsetMethod) || HttpMethod.POST.name().equals(requsetMethod)) {
    80. String parsams = argsArrayToString(joinPoint.getArgs());
    81. operLog.setOperParam(StringUtils.substring(parsams,0,2000));
    82. } else {
    83. Map<?,?> paramsMap = (Map<?,?>) ServletUtils.getRequest().getAttribute(HandlerMapping.URI_TEMPLATE_VARIABLES_ATTRIBUTE);
    84. operLog.setOperParam(StringUtils.substring(paramsMap.toString(),0,2000));
    85. }
    86. }
    87. /**
    88. * 参数拼装
    89. */
    90. private String argsArrayToString(Object[] paramsArray) {
    91. String params = "";
    92. if (paramsArray != null && paramsArray.length > 0) {
    93. for (Object object : paramsArray) {
    94. // 不为空 并且是不需要过滤的 对象
    95. if (StringUtils.isNotNull(object) && !isFilterObject(object)) {
    96. Object jsonObj = JSON.toJSON(object);
    97. params += jsonObj.toString() + " ";
    98. }
    99. }
    100. }
    101. return params.trim();
    102. }
    103. /**
    104. * 判断是否需要过滤的对象。
    105. * @param object 对象信息。
    106. * @return 如果是需要过滤的对象,则返回true;否则返回false。
    107. */
    108. @SuppressWarnings("rawtypes")
    109. public boolean isFilterObject(final Object object) {
    110. Class<?> clazz = object.getClass();
    111. if (clazz.isArray()) {
    112. return clazz.getComponentType().isAssignableFrom(MultipartFile.class);
    113. } else if (Collection.class.isAssignableFrom(clazz)) {
    114. Collection collection = (Collection) object;
    115. for (Object value : collection) {
    116. return value instanceof MultipartFile;
    117. }
    118. } else if (Map.class.isAssignableFrom(clazz)) {
    119. Map map = (Map) object;
    120. for (Object value : map.entrySet()) {
    121. Map.Entry entry = (Map.Entry) value;
    122. return entry.getValue() instanceof MultipartFile;
    123. }
    124. }
    125. return object instanceof MultipartFile || object instanceof HttpServletRequest
    126. || object instanceof HttpServletResponse || object instanceof BindingResult;
    127. }
    128. }
    129. 复制代码

    4. 标注在接口上

    将自定义注解标注在需要记录操作日志的接口上,代码如下(示例):

    1. @Log(title = "代码生成", businessType = BusinessType.GENCODE)
    2. @ApiOperation(value = "批量生成代码")
    3. @GetMapping("/download/batch")
    4. public void batchGenCode(HttpServletResponse response, String tables) throws IOException {
    5. String[] tableNames = Convert.toStrArray(tables);
    6. byte[] data = genTableService.downloadCode(tableNames);
    7. genCode(response, data);
    8. }
    9. 复制代码

    5. 实现的效果

    执行相关操作就会记录日志,记录了一些基础信息存在数据表里。


    总结

    好了,以上就是本篇文章的主要内容了,本文主要讲述了使用SpringAOP来实现操作日志的记录,欢迎评论区留言,说说你们的项目中是如何实现操作日志的。

    若觉得本文对您有帮助的话,还不忘点赞评论关注,支持一波哟~

  • 相关阅读:
    vue3 + vite 安装配置及项目创建
    三维数字沙盘大数据人工智能模拟对抗推演系统开发教程第一课
    利用fiddler正向代理前端请求到本地后端
    20年秦皇岛D - Exam Results(二分+思维,附易错数据)
    HTB-Heist(域、rc4_hmac、SeRestorePrivelege权限提权)
    代码随想录算法训练营day8 | 344.反转字符串、541. 反转字符串II、卡码网:54.替换数字、151.翻转字符串里的单词、卡码网:55.右旋转字符串
    载薄荷醇纳米多孔PS微球/LA57接枝纳米炭黑修饰/Ag纳米粒子/聚苯乙烯微球性能相关研究
    【练习八 结构体(强化)编程题7. 公共钥匙盒】
    JavaScript 解构的 5 个有趣用途
    第二章: 类加载子系统
  • 原文地址:https://blog.csdn.net/BASK2312/article/details/128014711