系统开发中我们常遇到要处理系统日志等信息的,在此我分享一篇 利用spring aop切面来异步添加日志的操作,其中用到了 队列和多线程,前面的博客有写。

第一步:创建log实体,根据自己业务而定,

package com.izhbg.typz.sso.audit.dto;
 
import java.util.Date;
 
import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.Id;
import javax.persistence.Table;
 
import org.hibernate.annotations.GenericGenerator;
 
@Entity
@Table(name="t_xt_audit_log")
public class AuditLog
{
    @Id
    @GenericGenerator(name = "paymentableGenerator", strategy = "assigned")   
    private String id;
    private String description;
    private String method;
    private Integer type;
    @Column(name="request_ip")
    private String requestIp;
    @Column(name="exception_code")
    private String exceptionCode;
    @Column(name="exception_detail")
    private String exceptionDetail;
    private String params;
    @Column(name="create_by")
    private String createBy;
    @Column(name="create_date")
    private Date createDate;
    @Column(name="app_id")
    private String appId;
    public String getId()
    {
        return id;
    }
    public void setId(String id)
    {
        this.id = id;
    }
    public String getDescription()
    {
        return description;
    }
    public void setDescription(String description)
    {
        this.description = description;
    }
    public String getMethod()
    {
        return method;
    }
    public void setMethod(String method)
    {
        this.method = method;
    }
    public Integer getType()
    {
        return type;
    }
    public void setType(Integer type)
    {
        this.type = type;
    }
    public String getRequestIp()
    {
        return requestIp;
    }
    public void setRequestIp(String requestIp)
    {
        this.requestIp = requestIp;
    }
    public String getExceptionCode()
    {
        return exceptionCode;
    }
    public void setExceptionCode(String exceptionCode)
    {
        this.exceptionCode = exceptionCode;
    }
    public String getExceptionDetail()
    {
        return exceptionDetail;
    }
    public void setExceptionDetail(String exceptionDetail)
    {
        this.exceptionDetail = exceptionDetail;
    }
    public String getParams()
    {
        return params;
    }
    public void setParams(String params)
    {
        this.params = params;
    }
    public String getCreateBy()
    {
        return createBy;
    }
    public void setCreateBy(String createBy)
    {
        this.createBy = createBy;
    }
    public Date getCreateDate()
    {
        return createDate;
    }
    public void setCreateDate(Date createDate)
    {
        this.createDate = createDate;
    }
    public String getAppId()
    {
        return appId;
    }
    public void setAppId(String appId)
    {
        this.appId = appId;
    }
    
    
}

第二步:创建日志的存放队列
package com.izhbg.typz.sso.audit.component;
 
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.concurrent.TimeUnit;
 
import org.springframework.stereotype.Component;
 
import com.izhbg.typz.sso.audit.dto.AuditLog;
 
 
@Component
public class AuditLogQueue {
    private BlockingQueue<AuditLog> blockingQueue = new LinkedBlockingQueue<AuditLog>();
 
    public void add(AuditLog auditLog) {
        blockingQueue.add(auditLog);
    }
 
    public AuditLog poll() throws InterruptedException {
        return blockingQueue.poll(1, TimeUnit.SECONDS);
    }
}
第三步:创建日志队列的 消费者
package com.izhbg.typz.sso.audit.component;
 
import java.util.ArrayList;
import java.util.List;
 
import javax.annotation.PostConstruct;
import javax.annotation.PreDestroy;
import javax.annotation.Resource;
 
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.stereotype.Component;
 
import com.izhbg.typz.sso.audit.dto.AuditLog;
import com.izhbg.typz.sso.audit.service.AuditLogService;
 
/**
 * 
* @ClassName: AuditLogConsumer 
* @Description: 日志的保存线程
* @author caixl 
* @date 2016-5-11 上午11:23:12 
*
 */
@Component
public class AuditLogConsumer  implements Runnable{
 
    private static Logger logger = LoggerFactory.getLogger(AuditLogConsumer.class);
    public static final int DEFAULT_BATCH_SIZE = 64;
    private AuditLogQueue auditLogQueue;
    private AuditLogService auditLogService;
    private int batchSize = DEFAULT_BATCH_SIZE;
    private boolean active = true;
    private Thread thread;
    
    @PostConstruct
    public void init() {
        thread = new Thread(this);
        thread.start();
    }
 
    @PreDestroy
    public void close() {
        active = false;
    }
 
    public void run() {
        while (active) {
            execute();
        }
    }
    
    public void execute() {
        List<AuditLog> auditDtos = new ArrayList<AuditLog>();
 
        try {
            int size = 0;
 
            while (size < batchSize) {
                AuditLog auditLog = auditLogQueue.poll();
 
                if (auditLog == null) {
                    break;
                }
 
                auditDtos.add(auditLog);
                size++;
            }
        } catch (Exception ex) {
            logger.info(ex.getMessage(), ex);
        }
 
        if (!auditDtos.isEmpty()) {
<span style="white-space:pre">        </span>//此处替换成自己的 service 处理即可,service就不贴了。
            auditLogService.batchLog(auditDtos);
        }
    }
 
    @Resource
    public void setAuditLogQueue(AuditLogQueue auditLogQueue) {
        this.auditLogQueue = auditLogQueue;
    }
    @Resource
    public void setAuditLogService(AuditLogService auditLogService) {
        this.auditLogService = auditLogService;
    }
 
    public void setBatchSize(int batchSize) {
        this.batchSize = batchSize;
    }
    
    
 
}

第四步:创建 controller层 自定义注解
package com.izhbg.typz.sso.annotation;
 
import java.lang.annotation.Documented;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
 
 
/**
 *     
* @ClassName: SystemControllerLog 
* @Description: 自定义注解 拦截Controller  
* @author caixl 
* @date 2016-5-10 下午2:16:12 
*
 */
@Target({ElementType.PARAMETER, ElementType.METHOD})    
@Retention(RetentionPolicy.RUNTIME)    
@Documented    
public  @interface SystemControllerLog {    
    String description()  default "";    
}    
第五步:创建service层自定义注解
package com.izhbg.typz.sso.annotation;
 
import java.lang.annotation.Documented;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
/**
 * 
* @ClassName: SystemServiceLog 
* @Description: 自定义注解 拦截service  
* @author caixl 
* @date 2016-5-10 下午2:17:12 
*
 */
@Target({ElementType.PARAMETER, ElementType.METHOD})    
@Retention(RetentionPolicy.RUNTIME)    
@Documented    
public  @interface SystemServiceLog {    
    String description()  default "";    
}

第六步:创建切点
package com.izhbg.typz.sso.annotation;
 
import java.lang.reflect.Method;
import java.util.Date;
 
import javax.annotation.Resource;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpSession;
 
import net.sf.json.util.JSONUtils;
 
import org.apache.log4j.Logger;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.stereotype.Component;
import org.springframework.web.context.request.RequestContextHolder;
import org.springframework.web.context.request.ServletRequestAttributes;
 
import com.izhbg.typz.base.util.CommonUtil;
import com.izhbg.typz.base.util.IdGenerator;
import com.izhbg.typz.sso.audit.component.AuditLogQueue;
import com.izhbg.typz.sso.audit.dto.AuditLog;
import com.izhbg.typz.sso.audit.manager.AuditLogManager;
import com.izhbg.typz.sso.util.SpringSecurityUtils;
/**
 * 
* @ClassName: AuditLogAspect 
* @Description: 切点类
* @author caixl 
* @date 2016-5-10 下午2:50:34 
*
 */
@Aspect    
@Component
public class AuditLogAspect
{
     private  AuditLogQueue auditLogQueue;
    //本地异常日志记录对象    
     private  static  final Logger logger = Logger.getLogger(AuditLogManager.class); 
     
     //Service层切点    
     @Pointcut("@annotation(com.izhbg.typz.sso.annotation.SystemServiceLog)")    
     public  void serviceAspect() {    
     }
     //Controller层切点    
     @Pointcut("@annotation(com.izhbg.typz.sso.annotation.SystemControllerLog)")    
     public  void controllerAspect() {    
     } 
     
     @Before("controllerAspect()")    
     public  void doBefore(JoinPoint joinPoint) { //@ TODO 异步线程处理 
         HttpServletRequest request = ((ServletRequestAttributes) RequestContextHolder.getRequestAttributes()).getRequest();
         HttpSession session = request.getSession();    
         //读取session中的用户    
         //请求的IP    
         String ip = CommonUtil.getIpAddress(request);
        //获取用户请求方法的参数并序列化为JSON格式字符串    
         String params = "";    
          if (joinPoint.getArgs() !=  null && joinPoint.getArgs().length > 0) {    
              for ( int i = 0; i < joinPoint.getArgs().length; i++) {    
                 params += JSONUtils.valueToString(joinPoint.getArgs()[i]) + ";";    
             }    
         }
         try {    
            //*========控制台输出=========*//    
            System.out.println("=====前置通知开始=====");    
            System.out.println("请求方法:" + (joinPoint.getTarget().getClass().getName() + "." + joinPoint.getSignature().getName() + "()"));    
            System.out.println("方法描述:" + getControllerMethodDescription(joinPoint));    
            System.out.println("请求人:" + SpringSecurityUtils.getCurrentUsername());    
            System.out.println("请求IP:" + ip);    
            //*========数据库日志=========*//    
            AuditLog log = new AuditLog();
            log.setId(IdGenerator.getInstance().getUniqTime()+"");
            log.setDescription(getControllerMethodDescription(joinPoint));    
            log.setMethod((joinPoint.getTarget().getClass().getName() + "." + joinPoint.getSignature().getName() + "()"));    
            log.setType(0);    
            log.setRequestIp(ip);    
            log.setExceptionCode( null);    
            log.setExceptionDetail( null);    
            log.setParams( params);    
            log.setCreateBy(SpringSecurityUtils.getCurrentUsername()+"("+SpringSecurityUtils.getCurrentUserId()+")");    
            log.setCreateDate(new Date());    
            log.setAppId(SpringSecurityUtils.getCurrentUserAppId());
            //保存数据库    
            auditLogQueue.add(log);   
            System.out.println("=====前置通知结束=====");    
        }  catch (Exception e) {    
            //记录本地异常日志    
            logger.error("==前置通知异常==");    
        }    
     }
     
     
    /**
     * 获取注解中对方法的描述信息 用于service层注解
     * 
     * @param joinPoint
     *            切点
     * @return 方法描述
     * @throws Exception
     */
    public static String getServiceMthodDescription(JoinPoint joinPoint) throws Exception
    {
        String targetName = joinPoint.getTarget().getClass().getName();
        String methodName = joinPoint.getSignature().getName();
        Object[] arguments = joinPoint.getArgs();
        Class targetClass = Class.forName(targetName);
        Method[] methods = targetClass.getMethods();
        String description = "";
        for (Method method : methods)
        {
            if (method.getName().equals(methodName))
            {
                Class[] clazzs = method.getParameterTypes();
                if (clazzs.length == arguments.length)
                {
                    description = method.getAnnotation(SystemServiceLog.class).description();
                    break;
                }
            }
        }
        return description;
    }
 
    /**
     * 获取注解中对方法的描述信息 用于Controller层注解
     * 
     * @param joinPoint
     *            切点
     * @return 方法描述
     * @throws Exception
     */
    public static String getControllerMethodDescription(JoinPoint joinPoint) throws Exception
    {
        String targetName = joinPoint.getTarget().getClass().getName();
        String methodName = joinPoint.getSignature().getName();
        Object[] arguments = joinPoint.getArgs();
        Class targetClass = Class.forName(targetName);
        Method[] methods = targetClass.getMethods();
        String description = "";
        for (Method method : methods)
        {
            if (method.getName().equals(methodName))
            {
                Class[] clazzs = method.getParameterTypes();
                if (clazzs.length == arguments.length)
                {
                    description = method.getAnnotation(SystemControllerLog.class).description();
                    break;
                }
            }
        }
        return description;
    }
    @Resource
    public void setAuditLogQueue(AuditLogQueue auditLogQueue) {
        this.auditLogQueue = auditLogQueue;
    }
    
    
    
    
 
}
</pre><pre name="code" class="java">
使用方式

@RedisCache(type=TXtYh.class,fieldKey="parameterMap.yhId")
    @RequestMapping("user-edit")
    @SystemControllerLog(description = "编辑用户")
    public String userEdit(@RequestParam Map<String, Object> parameterMap, Model model) throws Exception{
        
        String yhId= StringUtils.getString(parameterMap.get("yhId"));
        String currentAppId= StringUtils.getString(parameterMap.get("currentAppId"));
        TXtYh user = null;
        if(StringHelper.isNotEmpty(yhId))
            user = tXtYhService.findByYhId(yhId);
        String result = tXtJgService.getJgsJSON(currentAppId);
        model.addAttribute("user", user);
        model.addAttribute("result", result);
        model.addAttribute("currentAppId", currentAppId);
        
        List<TXtYy> tXtYyList = tXtYyService.queryAll();
        model.addAttribute("txtYy", tXtYyList);
        return "admin/guser/getguser";
    }

处理完成
--------------------- 
作者:月无心137 
来源:CSDN 
原文:https://blog.csdn.net/cai_chinasoft/article/details/51670160 
版权声明:本文为博主原创文章,转载请附上博文链接!

【经典】Spring aop切面实现异步添加日志—完整版相关推荐

  1. spring aop 切面添加日志

    这是一个非常简单的spring aop切面添加日志的程序,下面来看一下这个程序 1.程序使用jar包 2.切面类LoggingAspect.java package com.cailei.aop.as ...

  2. spring AOP切面日志

    spring AOP切面日志 导入依赖 <properties><fastjson.version>1.2.49</fastjson.version> </p ...

  3. spring AOP自定义注解方式实现日志管理

    转:spring AOP自定义注解方式实现日志管理 今天继续实现AOP,到这里我个人认为是最灵活,可扩展的方式了,就拿日志管理来说,用Spring AOP 自定义注解形式实现日志管理.废话不多说,直接 ...

  4. Spring AOP 切面@Around注解的具体使用

    @Around注解可以用来在调用一个具体方法前和调用后来完成一些具体的任务. 比如我们想在执行controller中方法前打印出请求参数,并在方法执行结束后来打印出响应值,这个时候,我们就可以借助于@ ...

  5. Spring AOP切面的时候参数的传递

    Spring AOP切面的时候参数的传递 Xml: <?xml version="1.0" encoding="UTF-8"?> <beans ...

  6. 日志管理(spring AOP切面拦截)

    **最近,在写日志管理的东西呢,用了几种方法试,还是拦截比较好用,直接截下来传入到数据库中存储. 写的时候,真是头疼啊. 哦~对了,忘了说了,不需要用到Dao层哦. 首先,来一个POJO实体类吧,我看 ...

  7. Spring AOP 实现业务和异常日志记录实战

    1 业务需求:今日,公司要求对操作的业务和日志统一做处理,需要把业务表数据相关信息存入日志表中,比如表名,方法名,业务id,操作操作时间modifyTIme等等. 除了在业务主动插入日志数据之外,有个 ...

  8. SpringBoot2.0 基础案例(11):配置AOP切面编程,解决日志记录业务

    本文源码 GitHub地址:知了一笑 https://github.com/cicadasmile/spring-boot-base 一.AOP切面编程 1.什么是AOP编程 在软件业,AOP为Asp ...

  9. Spring AOP切面使用详细解析

    相关文章: SpringBoot AOP切面的使用 一步一步手绘Spring AOP运行时序图(Spring AOP 源码分析) 架构师系列内容:架构师学习笔记(持续更新)) Spring AOP 应 ...

最新文章

  1. 【SICP练习】143 练习3.81
  2. The encryption certificate of the relying party trust identified by thumbprint is not valid
  3. OpenCV中函数imread的参数flags的枚举值(可设置值)及其具体意义
  4. python qtablewedgit_PyQt5-高级控件使用(QTableWidget)
  5. pythontcp_TCP编程
  6. spark从入门到精通spark内存管理详解- 堆内堆外内存管理
  7. 数据结构与算法之KMP算法
  8. 45-机器设计问题(深搜)
  9. centos7安装DHCP服务器
  10. (22)Vue.js 综合案例:TodoMVC
  11. 小程序组件报错Uncaught TypeError: Cannot read property 'name' of undefined
  12. 【linux】-bash: vim: 未找到命令
  13. 前后端通吃,vue大全Mark一下
  14. opencv codebook
  15. 支付宝小程序开发练习,显示自定义二维码(四)
  16. 基于Go语言Echo+Vue+ElementUI的OA办公系统
  17. Codeforces Round #429 (Div. 2) B. Godsend
  18. mongo按季度统计_2020年第一季度|我国DDoS攻击资源季度分析报告
  19. tlwdr5660间歇性掉线_tplink路由器频繁掉线的原因及解决办法
  20. 赵小楼《天道》《遥远的救世主》深度解析(48)为什么韩楚风说丁元英是“元英害怕女人是怕到骨子里了”?

热门文章

  1. 破解烽火移动HG6201M光猫 以获取超级管理密码 打开无线网
  2. 【PWA】web推送技术
  3. Java实现ChatRoom
  4. 如果离开一线城市,你会选择如何开始
  5. 铁路牵引变电所无人化改造方案 辅助监控系统
  6. H5唤醒本地APP(谷歌地图、百度地图等相通)
  7. 自动提交脚本python_百度收录自动化提交脚本 - python
  8. 《数据包是如何在网络中传输的》TODO
  9. unityUI插件整理
  10. 基于二分查找的抽签游戏算法的优化