如何写一个日志注解

1、注解

首先写一个注解

@Target({ElementType.METHOD, ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface AuditLog {
    /**
     * 操作信息
     */
    String operation();
}

2、写一个日志实体类

该类用来记录日志信息

@Setter
@Getter
public class Audit {
    /**
     * 操作时间
     */
    private LocalDateTime timestamp;
    /**
     * 应用名
     */
    private String applicationName;
    /**
     * 类名
     */
    private String className;
    /**
     * 方法名
     */
    private String methodName;
    /**
     * 用户id
     */
    private String userId;
    /**
     * 用户名
     */
    private String userName;
    /**
     * 租户id
     */
    private String clientId;
    /**
     * 操作信息
     */
    private String operation;
}

3、写一个切面类

@Slf4j
@Aspect
public class AuditLogAspect {
    @Value("${spring.application.name}")
    private String applicationName;
    
    private IAuditService auditService;

	//这里如果你的日志服务实现类注入到了spring容器中,spring在创建该类时会自动注入
    public AuditLogAspect(IAuditService auditService) {
        this.auditService = auditService;
    }

    /**
     * 用于SpEL表达式解析.
     */
    private SpelExpressionParser spelExpressionParser = new SpelExpressionParser();
    /**
     * 用于获取方法参数定义名字.
     */
    private DefaultParameterNameDiscoverer nameDiscoverer = new DefaultParameterNameDiscoverer();

    @Around("@within(auditLog) || @annotation(auditLog)")
    public void beforeMethod(ProceedingJoinPoint joinPoint, AuditLog auditLog) {

        if (auditLog == null) {
            // 获取类上的注解
            auditLog = joinPoint.getTarget().getClass().getDeclaredAnnotation(AuditLog.class);
        }
        Audit audit = getAudit(auditLog, joinPoint);
        auditService.save(audit);
        
    }

    /**
     * 解析spEL表达式
     */
    private String getValBySpEL(String spEL, MethodSignature methodSignature, Object[] args) {
        //获取方法形参名数组
        String[] paramNames = nameDiscoverer.getParameterNames(methodSignature.getMethod());
        if (paramNames != null && paramNames.length > 0) {
            Expression expression = spelExpressionParser.parseExpression(spEL);
            // spring的表达式上下文对象
            EvaluationContext context = new StandardEvaluationContext();
            // 给上下文赋值
            for(int i = 0; i < args.length; i++) {
                context.setVariable(paramNames[i], args[i]);
            }
            return expression.getValue(context).toString();
        }
        return null;
    }

    /**
     * 构建审计对象
     */
    private Audit getAudit(AuditLog auditLog, JoinPoint joinPoint) {
        Audit audit = new Audit();
        audit.setTimestamp(LocalDateTime.now());
        audit.setApplicationName(applicationName);

        MethodSignature methodSignature = (MethodSignature)joinPoint.getSignature();
        audit.setClassName(methodSignature.getDeclaringTypeName());
        audit.setMethodName(methodSignature.getName());

        ServletRequestAttributes attributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();
        HttpServletRequest request = attributes.getRequest();
        String userId = request.getHeader("x-userid-header");
        String userName = request.getHeader("x-user-header");
        String clientId = request.getHeader("x-tenant-header");
        audit.setUserId(userId);
        audit.setUserName(userName);
        audit.setClientId(clientId);

        String operation = auditLog.operation();
        if (operation.contains("#")) {
            //获取方法参数值
            Object[] args = joinPoint.getArgs();
            operation = getValBySpEL(operation, methodSignature, args);
        }
        audit.setOperation(operation);

        return audit;
    }
}

4、写一个日志服务类

public interface IAuditService {
    void save(Audit audit);
}

@Slf4j
@Service
public class LoggerAuditServiceImpl implements IAuditService {
    private static final String MSG_PATTERN = "{}|{}|{}|{}|{}|{}|{}|{}";

    /**
     * 格式为:{时间}|{应用名}|{类名}|{方法名}|{用户id}|{用户名}|{租户id}|{操作信息}
     * 例子:2020-02-04 09:13:34.650|user-center|com.central.user.controller.SysUserController|saveOrUpdate|1|admin|webApp|新增用户:admin
     */
    @Override
    public void save(Audit audit) {
        log.debug(MSG_PATTERN
                , audit.getTimestamp().format(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss.SSS"))
                , audit.getApplicationName(), audit.getClassName(), audit.getMethodName()
                , audit.getUserId(), audit.getUserName(), audit.getClientId()
                , audit.getOperation());
    }
}

这里我将其记录到日志文件并打印到控制台了,如果想要持久化到数据库,可以对IAuditService 进行数据库操作的实现。

  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在Java中,我们可以使用注解来为类、方法、变量等元素添加一些元数据信息。下面是一个简单的日志注解的示例: ```java import java.lang.annotation.*; @Target(ElementType.METHOD) // 该注解只能用于方法 @Retention(RetentionPolicy.RUNTIME) // 该注解在运行时保留 public @interface Log { String value() default ""; // 日志信息 } ``` 通过上述代码,我们定义了一个名为Log的注解,它只能用于方法上,并且在运行时保留。该注解包含一个名为value的属性,用来指定日志信息。 接下来,我们可以在需要打印日志的方法上使用该注解,例如: ```java public class Demo { @Log("执行了方法A") public void methodA() { // 方法A的实现 } } ``` 在上述示例中,我们在方法methodA上使用了Log注解,并指定了日志信息为“执行了方法A”。接下来,我们可以在方法执行前后打印日志: ```java public class LogUtil { public static void log(Method method) { Log log = method.getAnnotation(Log.class); if (log != null) { System.out.println(log.value() + " 开始执行..."); } // 执行方法 try { method.invoke(); } catch (Exception e) { e.printStackTrace(); } if (log != null) { System.out.println(log.value() + " 执行结束..."); } } } ``` 在上述代码中,我们定义了一个名为LogUtil的工具类,其中log方法接受一个Method类型的参数,表示需要打印日志的方法。我们首先通过getAnnotation方法获取该方法上的Log注解,如果存在,则打印日志信息并执行该方法。方法执行完毕后,再次检查Log注解是否存在,如果存在,则打印日志信息表示方法执行结束。 最后,我们可以在调用需要打印日志的方法时,使用LogUtil工具类来实现日志打印: ```java public class Main { public static void main(String[] args) { Demo demo = new Demo(); Method method = Demo.class.getMethod("methodA"); LogUtil.log(method); } } ``` 在上述代码中,我们首先创建了一个Demo对象,然后通过反射获取methodA方法,并将其传入LogUtil.log方法中。LogUtil.log方法会根据该方法上的Log注解来打印日志信息。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值