SpringBoot使用自定义注解结合AOP完成日志收集

一、创建一个自定义注解

     注解中包含了模块的名称以及具体操作,可以根据自己的需求进行修改或者扩展。

@Target({ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
public @interface OperationLogAnnotation {

    /**
     * 模块名称
     */
    String module();

    /**
     * 具体操作
     */
    String operate();

    /**
     * 备注
     */
    String remarks() default "";
}

二、创建切面类

@Aspect
@Component("com.*")
@Slf4j
public class OperationLogAspect {

    @Autowired
    private SysLogService sysLogService;

    // 配置切入点(以OperationLog注解为标志)
    @Pointcut("@annotation(com.app.appweb.anno.OperationLogAnnotation)")
    public void logPointCut() {

    }

    /**
     * 后置通知 用于拦截操作,在方法返回后执行
     *
     * @param joinPoint 切点
     */
    @AfterReturning(pointcut = "logPointCut()")
    public void doAfter(JoinPoint joinPoint) {
        handleLog(joinPoint, null);
    }

    /**
     * 拦截异常操作,有异常时执行
     */
    @AfterThrowing(value = "logPointCut()", throwing = "e")
    public void doAfter(JoinPoint joinPoint, Exception e) {
        handleLog(joinPoint, e);
    }

    private void handleLog(JoinPoint joinPoint, Exception e) {
        try {
            // 获得注解
            OperationLogAnnotation controllerLog = getAnnotationLog(joinPoint);
            if (controllerLog == null) {
                return;
            }
            // 模块详情
            Object module = controllerLog.module();
            // 具体操作
            Object operate = controllerLog.operate();
            // 访问类名
            String className = joinPoint.getTarget().getClass().getName();
            // 访问方法名
            String methodName = joinPoint.getSignature().getName();
            //参数
            Object[] params = joinPoint.getArgs();
            String param = "";

            for (int i = 0; i < params.length; i++) {
                param = param + "," + params[i] + "";
            }
            //RequestContextHolder:持有上下文的Request容器,获取到当前请求的request
            ServletRequestAttributes servletRequestAttributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();
            HttpServletRequest httpServletRequest = servletRequestAttributes.getRequest();
            //获取请求URL
            String path = httpServletRequest.getRequestURI();
           
            // 保存日志
            SysLog sysLog = new SysLog();
            sysLog.setOperation(operate.toString());
            sysLog.setMethod(methodName);
            sysLog.setPath(path);
            sysLog.setParam(param);
            sysLogService.insert(sysLog);
             //保存日志完成
        } catch (Exception ex) {
            // 异常日志输出
            log.error("异常信息:{}", ex.getMessage());
            ex.printStackTrace();
        }
    }

    /**
     * @Description 判断方法是否存在注解,如果存在注解则返回注解对象
     * @Date 2019/7/26 11:27
     * @Param [joinPoint]
     * @Exception
     */
    private OperationLogAnnotation getAnnotationLog(JoinPoint joinPoint) throws Exception {
        Signature signature = joinPoint.getSignature();
        MethodSignature methodSignature = (MethodSignature) signature;
        Method method = methodSignature.getMethod();
        if (method != null) {
            return method.getAnnotation(OperationLogAnnotation.class);
        }
        return null;
    }

}

三、在控制器的controller方法添加注解

@OperationLogAnnotation(module = "设置模块", operate = "新增系统用户")

将以上注解添加到用户新增的controller方法上即可。

 

你可能感兴趣的:(SpringBoot框架,Java)