经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » Java相关 » Spring Boot » 查看文章
Java?SpringBoot项目如何优雅的实现操作日志记录
来源:jb51  时间:2022/8/2 16:46:03  对本文有异议

前言

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

通常就是使用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>

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. */
  9. String title() default "";
  10.  
  11. /**
  12. * 功能
  13. */
  14. public BusinessType businessType() default BusinessType.OTHER;
  15.  
  16. /**
  17. * 是否保存请求的参数
  18. */
  19. public boolean isSaveRequestData() default true;
  20.  
  21. /**
  22. * 是否保存响应的参数
  23. */
  24. public boolean isSaveResponseData() default true;
  25. }

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

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. }

5. 实现的效果

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

总结

到此这篇关于Java SpringBoot项目如何优雅的实现操作日志记录的文章就介绍到这了,更多相关SpringBoot操作日志记录内容请搜索w3xue以前的文章或继续浏览下面的相关文章希望大家以后多多支持w3xue!

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号