暂无图片
暂无图片
暂无图片
暂无图片
暂无图片

基于SpringBoot使用AOP技术实现操作日志管理

码蚁在线 2021-03-19
462

操作日志对于程序员或管理员而言,可以快速定位到系统中相关的操作,而对于操作日志的管理的实现不能对正常业务实现进行影响,否则即不满足单一原则,也会导致后续代码维护困难,因此我们考虑使用AOP切面技术来实现对日志管理的实现。

文章大致内容:

1、基本概念

2、基本应用

3、日志管理实战

对这几部分理解了,会对AOP的应用应该很轻松。

一、基本概念

项目描述
Aspect(切面)跨越多个类的关注点的模块化,切面是通知和切点的结合。通知和切点共同定义了切面的全部内容——它是什么,在何时和何处完成其功能。事务处理和日志处理可以理解为切面
Join point(连接点)程序执行过程中的一个点,如方法的执行或异常的处理
Advice(通知)切面在特定连接点上采取的动作
Pointcut(切点)匹配连接点的断言。通知与切入点表达式相关联,并在切入点匹配的任何连接点上运行(例如,具有特定名称的方法的执行)。切入点表达式匹配的连接点概念是AOP的核心,Spring默认使用AspectJ切入点表达式语言
Introduction(引用)为类型声明其他方法或字段。Spring AOP允许您向任何建议的对象引入新的接口(和相应的实现)。例如,您可以使用介绍使bean实现IsModified接口,以简化缓存
Target object(目标)由一个或多个切面通知的对象。也称为“通知对象”。由于Spring AOP是通过使用运行时代理实现的,所以这个对象始终是代理对象
AOP proxy(代理)AOP框架为实现切面契约(通知方法执行等)而创建的对象。在Spring框架中,AOP代理是JDK动态代理或CGLIB代理
Weaving(织入)织入是将通知添加对目标类具体连接点上的过程,可以在编译时(例如使用AspectJ编译器)、加载时或运行时完成

Spring切面可以应用5种类型的通知:

  • 前置通知(Before):在目标方法被调用之前调用通知

  • 后置通知(After):在目标方法完成之后调用通知(无论是正常还是异常退出)

  • 返回通知(After-returning):在目标方法成功执行之后调用通知

  • 异常通知(After-throwing):在目标方法抛出异常后调用通知

  • 环绕通知(Around):通知包裹了被通知的方法,在被通知的方法调用之前和调用之后执行自定义的行为 其执行的顺序为: 后续的基本应用,会将 环绕通知前置通知后置通知返回通知异常通知进行实现,并演示其执行顺序。

二、基本应用

声明通知 大家可以将下面的代码复制出来,验证上面的执行顺序。

  1. @Aspect

  2. public class Test {

  3. private static int step = 0;


  4. @Pointcut("@annotation(com.chenyanwu.erp.erpframework.annotation.Log)") // the pointcut expression

  5. private void operation() {}


  6. @Before("operation()")

  7. public void doBeforeTask() {

  8. System.out.println(++step + " 前置通知");

  9. }


  10. @After("operation()")

  11. public void doAfterTask() {

  12. System.out.println(++step + " 后置通知");

  13. }


  14. @AfterReturning(pointcut = "operation()", returning = "retVal")

  15. public void doAfterReturnningTask(Object retVal) {

  16. System.out.println(++step + " 返回通知,返回值为:" + retVal.toString());

  17. }


  18. @AfterThrowing(pointcut = "operation()", throwing = "ex")

  19. public void doAfterThrowingTask(Exception ex) {

  20. System.out.println(++step + " 异常通知,异常信息为:" + ex.getMessage());

  21. }


  22. /**

  23. * 环绕通知需要携带ProceedingJoinPoint类型的参数

  24. * 环绕通知类似于动态代理的全过程ProceedingJoinPoint类型的参数可以决定是否执行目标方法

  25. * 且环绕通知必须有返回值,返回值即目标方法的返回值

  26. */

  27. //@Around("operation()")

  28. public Object doAroundTask(ProceedingJoinPoint pjp) {

  29. String methodname = pjp.getSignature().getName();

  30. Object result = null;

  31. try {

  32. // 前置通知

  33. System.out.println("目标方法" + methodname + "开始,参数为" + Arrays.asList(pjp.getArgs()));

  34. // 执行目标方法

  35. result = pjp.proceed();

  36. // 返回通知

  37. System.out.println("目标方法" + methodname + "执行成功,返回" + result);

  38. } catch (Throwable e) {

  39. // 异常通知

  40. System.out.println("目标方法" + methodname + "抛出异常: " + e.getMessage());

  41. }

  42. // 后置通知

  43. System.out.println("目标方法" + methodname + "结束");

  44. return result;

  45. }

  46. }

其中需要注意的是切入点:@Pointcut的表达式格式:

execution(modifiers-pattern? ret-type-pattern declaring-type-pattern? name-pattern(param-pattern)throws-pattern?) 括号中各个pattern分别表示:

  • 修饰符匹配(modifier-pattern?)

  • 返回值匹配(ret-type-pattern)可以为*表示任何返回值,全路径的类名等

  • 类路径匹配(declaring-type-pattern?)

  • 方法名匹配(name-pattern)可以指定方法名 或者 代表所有, set 代表以set开头的所有方法

  • 参数匹配((param-pattern))可以指定具体的参数类型,多个参数间用“,”隔开,各个参数也可以用“”来表示- 匹配任意类型的参数,如(String)表示匹配一个String参数的方法;(,String) 表示匹配有两个参数的方法,第一个参数可以是任意类型,而第二个参数是String类型;可以用(..)表示零个或多个任意参数

  • 异常类型匹配(throws-pattern?)

  • 其中后面跟着“?”的是可选项

示例:

1)execution(* (..))
//表示匹配所有方法
2)execution(public * com. savage.service.UserService.
(..))
//表示匹配com.savage.server.UserService中所有的公有方法
3)execution(* com.savage.server...(..))
//表示匹配com.savage.server包及其子包下的所有方法

三、日志管理实战

有了上面基本应用的理解,现在我们直接就贴代码:1、依赖的jar包

  1. <!-- aop依赖 -->

  2. <dependency>

  3. <groupId>org.springframework.boot</groupId>

  4. <artifactId>spring-boot-starter-aop</artifactId>

  5. </dependency>

2、自定义注解

  1. @Target(ElementType.METHOD)

  2. @Retention(RetentionPolicy.RUNTIME)

  3. public @interface Log {

  4. String value() default "";

  5. }

3、实现切面

  1. @Aspect

  2. @Order(5)

  3. @Component

  4. public class LogAspect {


  5. private Logger logger = LoggerFactory.getLogger(LogAspect.class);


  6. @Autowired

  7. private ErpLogService logService;


  8. @Autowired

  9. ObjectMapper objectMapper;


  10. private ThreadLocal<Date> startTime = new ThreadLocal<Date>();


  11. @Pointcut("@annotation(com.chenyanwu.erp.erpframework.annotation.Log)")

  12. public void pointcut() {


  13. }


  14. /**

  15. * 前置通知,在Controller层操作前拦截

  16. *

  17. * @param joinPoint 切入点

  18. */

  19. @Before("pointcut()")

  20. public void doBefore(JoinPoint joinPoint) {

  21. // 获取当前调用时间

  22. startTime.set(new Date());

  23. }


  24. /**

  25. * 正常情况返回

  26. *

  27. * @param joinPoint 切入点

  28. * @param rvt 正常结果

  29. */

  30. @AfterReturning(pointcut = "pointcut()", returning = "rvt")

  31. public void doAfter(JoinPoint joinPoint, Object rvt) throws Exception {

  32. handleLog(joinPoint, null, rvt);

  33. }


  34. /**

  35. * 异常信息拦截

  36. *

  37. * @param joinPoint

  38. * @param e

  39. */

  40. @AfterThrowing(pointcut = "pointcut()", throwing = "e")

  41. public void doAfter(JoinPoint joinPoint, Exception e) throws Exception {

  42. handleLog(joinPoint, e, null);

  43. }


  44. @Async

  45. private void handleLog(final JoinPoint joinPoint, final Exception e, Object rvt) throws Exception{

  46. // 获得注解

  47. Method method = getMethod(joinPoint);

  48. Log log = getAnnotationLog(method);

  49. if (log == null) {

  50. return;

  51. }

  52. Date now = new Date();

  53. // 操作数据库日志表

  54. ErpLog erpLog = new ErpLog();

  55. erpLog.setErrorCode(0);

  56. erpLog.setIsDeleted(0);

  57. // 请求信息

  58. HttpServletRequest request = ToolUtil.getRequest();

  59. erpLog.setType(ToolUtil.isAjaxRequest(request) ? "Ajax请求" : "普通请求");

  60. erpLog.setTitle(log.value());

  61. erpLog.setHost(request.getRemoteHost());

  62. erpLog.setUri(request.getRequestURI().toString());

  63. // erpLog.setHeader(request.getHeader(HttpHeaders.USER_AGENT));

  64. erpLog.setHttpMethod(request.getMethod());

  65. erpLog.setClassMethod(joinPoint.getSignature().getDeclaringTypeName() + "." + joinPoint.getSignature().getName());

  66. // 请求的方法参数值

  67. Object[] args = joinPoint.getArgs();

  68. // 请求的方法参数名称

  69. LocalVariableTableParameterNameDiscoverer u

  70. = new LocalVariableTableParameterNameDiscoverer();

  71. String[] paramNames = u.getParameterNames(method);

  72. if (args != null && paramNames != null) {

  73. StringBuilder params = new StringBuilder();

  74. params = handleParams(params, args, Arrays.asList(paramNames));

  75. erpLog.setParams(params.toString());

  76. }

  77. String retString = JsonUtil.bean2Json(rvt);

  78. erpLog.setResponseValue(retString.length() > 5000 ? JsonUtil.bean2Json("请求参数数据过长不与显示") : retString);

  79. if (e != null) {

  80. erpLog.setErrorCode(1);

  81. erpLog.setErrorMessage(e.getMessage());

  82. }

  83. Date stime = startTime.get();

  84. erpLog.setStartTime(stime);

  85. erpLog.setEndTime(now);

  86. erpLog.setExecuteTime(now.getTime() - stime.getTime());

  87. erpLog.setUsername(MySysUser.loginName());

  88. HashMap<String, String> browserMap = ToolUtil.getOsAndBrowserInfo(request);

  89. erpLog.setOperatingSystem(browserMap.get("os"));

  90. erpLog.setBrower(browserMap.get("browser"));

  91. erpLog.setId(IdUtil.simpleUUID());

  92. logService.insertSelective(erpLog);

  93. }


  94. /**

  95. * 是否存在注解,如果存在就获取

  96. */

  97. private Log getAnnotationLog(Method method) {

  98. if (method != null) {

  99. return method.getAnnotation(Log.class);

  100. }

  101. return null;

  102. }


  103. private Method getMethod(JoinPoint joinPoint) {

  104. Signature signature = joinPoint.getSignature();

  105. MethodSignature methodSignature = (MethodSignature) signature;

  106. Method method = methodSignature.getMethod();

  107. if (method != null) {

  108. return method;

  109. }

  110. return null;

  111. }


  112. private StringBuilder handleParams(StringBuilder params, Object[] args, List paramNames) throws JsonProcessingException {

  113. for (int i = 0; i < args.length; i++) {

  114. if (args[i] instanceof Map) {

  115. Set set = ((Map) args[i]).keySet();

  116. List list = new ArrayList();

  117. List paramList = new ArrayList<>();

  118. for (Object key : set) {

  119. list.add(((Map) args[i]).get(key));

  120. paramList.add(key);

  121. }

  122. return handleParams(params, list.toArray(), paramList);

  123. } else {

  124. if (args[i] instanceof Serializable) {

  125. Class<?> aClass = args[i].getClass();

  126. try {

  127. aClass.getDeclaredMethod("toString", new Class[]{null});

  128. // 如果不抛出NoSuchMethodException 异常则存在 toString 方法 ,安全的writeValueAsString ,否则 走 Object的 toString方法

  129. params.append(" ").append(paramNames.get(i)).append(": ").append(objectMapper.writeValueAsString(args[i]));

  130. } catch (NoSuchMethodException e) {

  131. params.append(" ").append(paramNames.get(i)).append(": ").append(objectMapper.writeValueAsString(args[i].toString()));

  132. }

  133. } else if (args[i] instanceof MultipartFile) {

  134. MultipartFile file = (MultipartFile) args[i];

  135. params.append(" ").append(paramNames.get(i)).append(": ").append(file.getName());

  136. } else {

  137. params.append(" ").append(paramNames.get(i)).append(": ").append(args[i]);

  138. }

  139. }

  140. }

  141. return params;

  142. }

  143. }

4、对应代码添加注解

  1. @Log("新增学生")

  2. @RequestMapping(value = "/create", method = RequestMethod.POST)

  3. @ResponseBody

  4. public ResultBean<String> create(@RequestBody @Validated ErpStudent item) {

  5. if(service.insertSelective(item) == 1) {

  6. // 插入

  7. insertErpSFamilyMember(item);

  8. return new ResultBean<String>("");

  9. }


  10. return new ResultBean<String>(ExceptionEnum.BUSINESS_ERROR, "新增学生异常!", "新增失败!", "");

  11. }

通过对业务进行操作后,会写入数据库,界面查询:

日志管理的完整的代码可以从git上获取:https://github.com/chyanwu/erp-framework


写作不易,坚持更难,如大家喜欢就帮忙推送给其他人!

更多精彩,更多技术请关注:码蚁在线(coding_online)


文章转载自码蚁在线,如果涉嫌侵权,请发送邮件至:contact@modb.pro进行举报,并提供相关证据,一经查实,墨天轮将立刻删除相关内容。

评论