spring AOP 实现日志写库

设计原则和思路:

  • 元注解方式结合AOP,灵活记录操作日志
  • 能够记录详细错误日志为运维提供支持
  • 日志记录尽可能减少性能影响

1.定义日志记录元注解

2.定义用于记录日志的实体类

  1. import java.io.Serializable;
  2. import com.leon.common.util.StringUtils;
  3. import com.fasterxml.jackson.annotation.JsonFormat;
  4. import java.util.Date;
  5. import java.util.Map;
  6. /**
  7. * 日志类-记录用户操作行为
  8. * @author lin.r.x
  9. *
  10. */
  11. public class Log implements Serializable{
  12. private static final long serialVersionUID = 1 L;
  13. private String logId; //日志主键
  14. private String type; //日志类型
  15. private String title; //日志标题
  16. private String remoteAddr; //请求地址
  17. private String requestUri; //URI
  18. private String method; //请求方式
  19. private String params; //提交参数
  20. private String exception; //异常
  21. @JsonFormat(pattern= "yyyy-MM-dd HH:mm:ss", timezone = "GMT+8")
  22. private Date operateDate; //开始时间
  23. private String timeout; //结束时间
  24. private String userId; //用户ID
  25.      private String resultParams;     //返回参数
  26. public String getLogId () {
  27. return StringUtils.isBlank(logId) ? logId : logId.trim();
  28. }
  29. public void setLogId (String logId) {
  30. this.logId = logId;
  31. }
  32.      public String getResultParams() {
  33.          return resultParams;
  34.     }
  35.      public void setResultParams(String resultParams) {
  36.          this.resultParams = resultParams;
  37.     }
  38. public String getType () {
  39. return StringUtils.isBlank(type) ? type : type.trim();
  40. }
  41. public void setType (String type) {
  42. this.type = type;
  43. }
  44. public String getTitle () {
  45. return StringUtils.isBlank(title) ? title : title.trim();
  46. }
  47. public void setTitle (String title) {
  48. this.title = title;
  49. }
  50. public String getRemoteAddr () {
  51. return StringUtils.isBlank(remoteAddr) ? remoteAddr : remoteAddr.trim();
  52. }
  53. public void setRemoteAddr (String remoteAddr) {
  54. this.remoteAddr = remoteAddr;
  55. }
  56. public String getRequestUri () {
  57. return StringUtils.isBlank(requestUri) ? requestUri : requestUri.trim();
  58. }
  59. public void setRequestUri (String requestUri) {
  60. this.requestUri = requestUri;
  61. }
  62. public String getMethod () {
  63. return StringUtils.isBlank(method) ? method : method.trim();
  64. }
  65. public void setMethod (String method) {
  66. this.method = method;
  67. }
  68. public String getParams () {
  69. return StringUtils.isBlank(params) ? params : params.trim();
  70. }
  71. public void setParams (String params) {
  72. this.params = params;
  73. }
  74. /**
  75. * 设置请求参数
  76. * @param paramMap
  77. */
  78. public void setMapToParams (Map<String, String[]> paramMap) {
  79. if (paramMap == null){
  80. return;
  81. }
  82. StringBuilder params = new StringBuilder();
  83. for (Map.Entry<String, String[]> param : ((Map<String, String[]>)paramMap).entrySet()){
  84. params.append(( "".equals(params.toString()) ? "" : "&") + param.getKey() + "=");
  85. String paramValue = (param.getValue() != null && param.getValue().length > 0 ? param.getValue()[ 0] : "");
  86. params.append(StringUtils.abbr(StringUtils.endsWithIgnoreCase(param.getKey(), "password") ? "" : paramValue, 100));
  87. }
  88. this.params = params.toString();
  89. }
  90. public String getException () {
  91. return StringUtils.isBlank(exception) ? exception : exception.trim();
  92. }
  93. public void setException (String exception) {
  94. this.exception = exception;
  95. }
  96. public Date getOperateDate () {
  97. return operateDate;
  98. }
  99. public void setOperateDate (Date operateDate) {
  100. this.operateDate = operateDate;
  101. }
  102. public String getTimeout () {
  103. return StringUtils.isBlank(timeout) ? timeout : timeout.trim();
  104. }
  105. public void setTimeout (String timeout) {
  106. this.timeout = timeout;
  107. }
  108. public String getUserId () {
  109. return StringUtils.isBlank(userId) ? userId : userId.trim();
  110. }
  111. public void setUserId (String userId) {
  112. this.userId = userId;
  113. }
  114. }

工具类代码:

3.定义日志AOP切面类(日志新增及修改其它业务操作不做介绍):

package com.isoftstone.api.common.log.aspect;

import java.lang.reflect.Method;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.Map;

import javax.servlet.http.HttpServletRequest;

import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.AfterReturning;
import org.aspectj.lang.annotation.AfterThrowing;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.aspectj.lang.reflect.MethodSignature;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.core.NamedThreadLocal;
import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor;
import org.springframework.stereotype.Component;

import com.isoftstone.api.common.config.security.UserUtils;
import com.isoftstone.api.common.log.target.SystemControllerLog;
import com.isoftstone.api.common.log.target.SystemServiceLog;
import com.isoftstone.api.common.log.utils.DateUtils;
import com.isoftstone.api.common.log.utils.UuidUtils;
import com.isoftstone.api.common.utils.JsonUtils;
import com.isoftstone.api.common.utils.redis.RedisUtil;
import com.isoftstone.api.log.entity.Log;
import com.isoftstone.api.log.service.LogService;
import com.isoftstone.api.system.entity.Users;


/**
* @author leon
* @createDate 2018年6月6日 上午10:33:35
* @version v1.0
* @classRemarks 日志切面类
*/
@Aspect
@Component
public class SystemLogAspect {
    
    private static Logger logger = LoggerFactory.getLogger(SystemLogAspect.class);
    
    private static final ThreadLocal<Date> beginTimeThreadLocal = new NamedThreadLocal<Date>("ThreadLocal beginTime");
    private static final ThreadLocal<Log> logThreadLocal =  new NamedThreadLocal<Log>("ThreadLocal log");

    private static final ThreadLocal<Users> currentUser=new NamedThreadLocal<>("ThreadLocal user");
    
    @Autowired(required=false)
    HttpServletRequest request;

    @Autowired
    ThreadPoolTaskExecutor threadPoolTaskExecutor;
    
    @Autowired
    LogService logService;
    
    @Autowired
    RedisUtil redisUtil;
    
    /**
     * Service层切点 
     */
    @Pointcut("@annotation(com.isoftstone.api.common.log.target.SystemServiceLog)")
    public void serviceAspect(){}
    
    /**
     * Controller层切点 注解拦截
     */
    @Pointcut("@annotation(com.isoftstone.api.common.log.target.SystemControllerLog)")
    public void controllerAspect(){}
    
    /*@Pointcut("execution(* com.isoftstone.api.*.controller.*.*(..))")
    public void controllerPointerCut(){}*/

    /**
     * 前置通知 用于拦截Controller层记录用户的操作的开始时间
     * @param joinPoint 切点
     * @throws InterruptedException 
     */
    @Before("controllerAspect()")
    public void doBefore(JoinPoint joinPoint) throws InterruptedException{
        
        logger.info("进入日志切面前置通知!!");
        Date beginTime=new Date();
        beginTimeThreadLocal.set(beginTime);//线程绑定变量(该数据只有当前请求的线程可见)  
        if (logger.isDebugEnabled()){//这里日志级别为debug
            logger.debug("开始计时: {}  URI: {}", new SimpleDateFormat("yyyy-MM-dd HH:mm:ss.SSS")
                .format(beginTime), request.getRequestURI());
        }

        //读取session中的用户 
        /*HttpSession session = request.getSession();       
        Users user = (Users) session.getAttribute("ims_user"); */   
        Users user=UserUtils.getUser(redisUtil);
        currentUser.set(user);

    }

    /**
     * 后置通知 用于拦截Controller层记录用户的操作
     * @param joinPoint 切点
     */
    @After("controllerAspect()")
    public void doAfter(JoinPoint joinPoint) {
        logger.info("进入日志切面后置通知!!");
        Users user = currentUser.get();
        if(user !=null){
            String title="";
            String type="info";                       //日志类型(info:入库,error:错误)
            String remoteAddr=request.getRemoteAddr();//请求的IP
            String requestUri=request.getRequestURI();//请求的Uri
            String method=request.getMethod();        //请求的方法类型(post/get)
            Map<String,String[]> params=request.getParameterMap(); //请求提交的参数
            
            try {
                title=getControllerMethodDescription2(joinPoint);
            } catch (Exception e) {
                e.printStackTrace();
            }    
            // 打印JVM信息。
            long beginTime = beginTimeThreadLocal.get().getTime();//得到线程绑定的局部变量(开始时间)  
            long endTime = System.currentTimeMillis();  //2、结束时间  
            if (logger.isDebugEnabled()){
                logger.debug("计时结束:{}  URI: {}  耗时: {}   最大内存: {}m  已分配内存: {}m  已分配内存中的剩余空间: {}m  最大可用内存: {}m",
                        new SimpleDateFormat("yyyy-MM-dd HH:mm:ss.SSS").format(endTime), 
                        request.getRequestURI(), 
                        DateUtils.formatDateTime(endTime - beginTime),
                        Runtime.getRuntime().maxMemory()/1024/1024, 
                        Runtime.getRuntime().totalMemory()/1024/1024, 
                        Runtime.getRuntime().freeMemory()/1024/1024, 
                        (Runtime.getRuntime().maxMemory()-Runtime.getRuntime().totalMemory()+Runtime.getRuntime().freeMemory())/1024/1024); 
            }
            logger.info("设置日志信息存储到表中!!");
            Log log=new Log();
            log.setLogId(UuidUtils.creatUUID());
            log.setTitle(title);
            log.setType(type);
            log.setRemoteAddr(remoteAddr);
            log.setRequestUri(requestUri);
            log.setMethod(method);
            log.setMapToParams(params);
            log.setException("无异常!");
            log.setUserId(user.getUsername());
            Date operateDate=beginTimeThreadLocal.get();
            log.setOperateDate(operateDate);
            log.setTimeout(DateUtils.formatDateTime(endTime - beginTime));

            //1.直接执行保存操作
            //this.logService.addLog(log);

            //2.优化:异步保存日志
            //new SaveLogThread(log, logService).start();

            //3.再优化:通过线程池来执行日志保存
            threadPoolTaskExecutor.execute(new SaveLogThread(log, logService));
            logThreadLocal.set(log);
        }

    }
    
    @AfterReturning(returning = "res", pointcut = "controllerAspect()")
    public void doAfterReturning(Object res) throws Throwable {
        // 处理完请求,返回内容
        logger.info("==========返回参数日志=========");
        logger.info("返回接口响应参数:"+JsonUtils.obj2JSON(res));
        Log log = logThreadLocal.get();
        if (log!=null) {
            log.setResultParams(JsonUtils.obj2JSON(res));
            logger.info("==========更新日志参数=========");
            new UpdateLogThread(log, logService).start();
        }
    }

    /**
     *  异常通知 记录操作报错日志
     * @param joinPoint
     * @param e
     */
    @AfterThrowing(pointcut = "controllerAspect()", throwing = "e")  
    public  void doAfterThrowing(JoinPoint joinPoint, Throwable e) {
        logger.info("进入日志切面异常通知!!");
        logger.info("异常信息:"+e.getMessage());
        Log log = logThreadLocal.get();
        if (log!=null) {
            log.setType("error");
            log.setException(e.toString());
            new UpdateLogThread(log, logService).start();
        }
    }

    /**
     * 获取注解中对方法的描述信息 用于service层注解
     * @param joinPoint切点
     * @return discription
     */
    public static String getServiceMthodDescription2(JoinPoint joinPoint) {
        MethodSignature signature = (MethodSignature) joinPoint.getSignature();
        Method method = signature.getMethod();
        SystemServiceLog serviceLog = method.getAnnotation(SystemServiceLog.class);
        String discription = serviceLog.description();
        return discription;
    }

    /**
     * 获取注解中对方法的描述信息 用于Controller层注解
     * 
     * @param joinPoint 切点
     * @return discription
     */
    public static String getControllerMethodDescription2(JoinPoint joinPoint) {
        MethodSignature signature = (MethodSignature) joinPoint.getSignature();
        Method method = signature.getMethod();
        SystemControllerLog controllerLog = method.getAnnotation(SystemControllerLog.class);
        String discription = controllerLog.description();
        return discription;
    }

    /**
     * 保存日志线程
     */
    private static class SaveLogThread implements Runnable {
        private Log log;
        private LogService logService;

        public SaveLogThread(Log log, LogService logService) {
            this.log = log;
            this.logService = logService;
        }

        @Override
        public void run() {
            logService.addLog(log);
        }
    }

    /**
     * 日志更新线程
     */
    private static class UpdateLogThread extends Thread {
        private Log log;
        private LogService logService;

        public UpdateLogThread(Log log, LogService logService) {
            super(UpdateLogThread.class.getSimpleName());
            this.log = log;
            this.logService = logService;
        }

        @Override
        public void run() {
            this.logService.putLog(log);
        }
    }

}

线程池配置代码:

package com.isoftstone.api.common.log.thread;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor;

/**
* @author leon
* @createDate 2018年6月6日 上午11:23:22
* @version v1.0
* @classRemarks 线程池配置
*/

@Configuration
@ConfigurationProperties(prefix="threadpool")
public class ExecutePoolConfiguration {
    
    private static Logger logger = LoggerFactory.getLogger(ExecutePoolConfiguration.class);

    /**
     * application.yml 配置方式
        threadpool:
          core-pool-size: 10
          max-pool-size: 20
          queue-capacity: 1000
          keep-alive-seconds: 300
     */
    @Value("${threadpool.core-pool-size}")
    private int corePoolSize;

    @Value("${threadpool.max-pool-size}")
    private int maxPoolSize;

    @Value("${threadpool.queue-capacity}")
    private int queueCapacity;

    @Value("${threadpool.keep-alive-seconds}")
    private int keepAliveSeconds;


    @Bean(name="threadPoolTaskExecutor")
    public ThreadPoolTaskExecutor threadPoolTaskExecutor(){
        
        ThreadPoolTaskExecutor pool = new ThreadPoolTaskExecutor();
        pool.setKeepAliveSeconds(keepAliveSeconds);
        pool.setCorePoolSize(corePoolSize);//核心线程池数
        pool.setMaxPoolSize(maxPoolSize); // 最大线程
        pool.setQueueCapacity(queueCapacity);//队列容量
        pool.setRejectedExecutionHandler(new java.util.concurrent.ThreadPoolExecutor.CallerRunsPolicy()); //队列满,线程被拒绝执行策略
        return pool;
    }

}

属性文件中添加下列配置:


4.spring boot配置扫描切面,开启@AspectJ注解的支持

5.使用范例LoginController方法中添加日志注解

6.运行效果

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Spring AOP是一个强大的框架,可以帮助我们实现各种切面,其中包括日志记录。下面是实现日志记录的步骤: 1. 添加Spring AOP依赖 在Maven或Gradle中添加Spring AOP依赖。 2. 创建日志切面 创建一个用于记录日志的切面。这个切面可以拦截所有需要记录日志的方法。在这个切面中,我们需要使用@Aspect注解来声明这是一个切面,并使用@Pointcut注解来定义哪些方法需要被拦截。 ```java @Aspect @Component public class LoggingAspect { @Pointcut("execution(* com.example.demo.service.*.*(..))") public void serviceMethods() {} @Around("serviceMethods()") public Object logServiceMethods(ProceedingJoinPoint joinPoint) throws Throwable { // 获取方法名,参数列表等信息 String methodName = joinPoint.getSignature().getName(); Object[] args = joinPoint.getArgs(); // 记录日志 System.out.println("Method " + methodName + " is called with args " + Arrays.toString(args)); // 执行方法 Object result = joinPoint.proceed(); // 记录返回值 System.out.println("Method " + methodName + " returns " + result); return result; } } ``` 在上面的代码中,我们使用了@Around注解来定义一个环绕通知,它会在拦截的方法执行前后执行。在方法执行前,我们记录了该方法的名称和参数列表,然后在方法执行后记录了该方法的返回值。 3. 配置AOPSpring的配置文件中配置AOP。首先,我们需要启用AOP: ```xml <aop:aspectj-autoproxy/> ``` 然后,我们需要将创建的日志切面添加到AOP中: ```xml <bean id="loggingAspect" class="com.example.demo.aspect.LoggingAspect"/> <aop:config> <aop:aspect ref="loggingAspect"> <aop:pointcut id="serviceMethods" expression="execution(* com.example.demo.service.*.*(..))"/> <aop:around method="logServiceMethods" pointcut-ref="serviceMethods"/> </aop:aspect> </aop:config> ``` 在上面的代码中,我们将创建的日志切面声明为一个bean,并将其添加到AOP中。我们还定义了一个切入点,并将其与日志切面的方法进行关联。 4. 测试 现在,我们可以测试我们的日志记录功能了。在我们的业务逻辑中,所有匹配切入点的方法都会被拦截,并记录它们的输入和输出。我们可以在控制台中看到这些日志信息。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值