zoukankan      html  css  js  c++  java
  • AOP行为日志

    最近新项目要记录行为日志,很久没有用AOP,研究了一下。

    废话不多说,先上个流程图:

    数据库日志表设计

    字段名称 字段类型 注释
    LOG_ID VARCHAR2(255)  
    LOG_LEVEL  NUMBER  日志级别
    START_TIME  DATE  开始时间
    RUN_TIME  NUMBER  运行时间(ms)
    OPERATION_MODULE  VARCHAR2(255)  被操作的模块
    OPERATION_UNIT  VARCHAR2(255)  被操作的单元
    OPERATION_TYPE  VARCHAR2(255)  操作类型
    OPERATION_DETAIL  VARCHAR2(500 CHAR)  操作详情
    USER_CODE  VARCHAR2(255)  用户编号
    USER_NAME  VARCHAR2(255)  用户名称

    注:数据库使用的Oracle

    JAVA端

    1、创建日志实体类

    import com.fasterxml.jackson.annotation.JsonFormat;
    import lombok.Data;
    
    import java.util.Date;
    
    @Data
    public class OperationLog {
        private String logId;
        private String userCode;
        private String userName;
        @JsonFormat(locale="zh", timezone="GMT+8", pattern="yyyy-MM-dd HH:mm:ss")
        private Date startTime;
        private Long runTime;
        private String operationUnit;
        private String operationType;
        private String operationDetail;
        private String operationModule;
        private Integer logLevel;
    }

    2、创建日志操作类型、单元、模块等枚举类

    (1)操作模块枚举类

    public enum OperationModule {
        /**
         * 被操作的模块
         */
        UNKNOWN("XX系统"),
        USER("用户模块"),
        PRODUCT("产品模块"),
        SALE("销售信息模块");
    
        private String value;
    
        public String getValue() {
            return value;
        }
    
        public void setValue(String value) {
            this.value = value;
        }
    
        OperationModule(String s) {
            this.value = s;
        }
    }

    (2)操作单元枚举类

    public enum OperationUnit {
        /**
         * 被操作的单元
         */
        UNKNOWN(""),
        /**
         * 用户模块
         */
        USER_INFO("用户信息"),
        USER_ROLE("用户角色"),
        USER_PERMISSION("用户权限"),
        /**
         * 产品模块
         */
        PRODUCT_INFO("产品信息"),
        PRODUCT_INV("产品库存"),
        /**
         * 销售信息模块
         */
        SALE_INFO("销售信息"),
        SALE_PLAN("销售计划");
    
    
        private String value;
    
        OperationUnit(String value) {
            this.value = value;
        }
    
        public String getValue() {
            return value;
        }
    
        public void setValue(String value) {
            this.value = value;
        }
    }

    (3)操作类型枚举类

    public enum OperationType {
        /**
         * 基本操作类型
         */
        UNKNOWN(""),
        LOGIN("登录"),
        INPUT("导入"),
        QUERY("查询"),
        EXPORT("导出"),
        DELETE("删除"),
        INSERT("插入"),
        UPDATE("更新"),
        /**
         * 用户
         */
        USER_SET_ROLE("设置用户角色"),
        USER_SET_PERMISSION("设置用户权限"),
        /**
         * 商品
         */
        PRODUCT_EXPORT_INFO("导出商品信息"),
        PRODUCT_SET_RANK("设置商品级别"),
        /**
         * 销售
         */
        SALE_EXPORT_INFO("导出销售信息"),
        SALE_SET_SALE_PLAN("设置销售计划");
    
        private String value;
    
        public String getValue() {
            return value;
        }
    
        public void setValue(String value) {
            this.value = value;
        }
    
        OperationType(String s) {
            this.value = s;
        }
    }

    3、创建日志注解

    import com.XXX.XXX.domin.OperationModule;
    import com.XXX.XXX.domin.OperationType;
    import com.XXX.XXX.domin.OperationUnit;
    
    import java.lang.annotation.*;
    
    @Documented //表明这个注解应该被 javadoc工具记录
    @Target({ElementType.METHOD})   //声明该注解作用于方法之上
    @Retention(RetentionPolicy.RUNTIME) //声明该注解不仅被保存到class文件中,jvm加载class文件之后,仍然存在
    public @interface OperationLogDetail {
    
        /**
         * 方法描述,可使用占位符获取参数:{{param}}
         */
        String operationDetail() default "";
    
        /**
         * 日志等级:1-9
         */
        int logLevel() default 1;
    
        /**
         * 操作类型(enum)
         */
        OperationType operationType() default OperationType.UNKNOWN;
    
        /**
         * 被操作的对象(此处使用enum)
         */
        OperationUnit operationUnit() default OperationUnit.UNKNOWN;
    
        /**
         * 被操作的系统模块(此处使用enum)
         */
        OperationModule operationModule() default OperationModule.UNKNOWN;
    
    }

    4、创建AOP方法,使用了环绕通知

    @Aspect     //表明该类是一个切面
    @Component  //实例化到spring容器中
    public class OperationLogAop {
    
        @Autowired
        private OperationLogService operationLogService;
        
        //表明切点在加了OperationLogDetail注解的方法
        @Pointcut("@annotation(com.topsports.adbuhuo.annotation.OperationLogDetail)")
        public void operationLog(){}
        
        //环绕通知
        @Around("operationLog()")
        public Object doAround(ProceedingJoinPoint joinPoint) throws Throwable {
            Object res = null;
            //获取系统当前时间
            long time = System.currentTimeMillis();
            try {
                //获取切入点(要记录日志的方法)的参数
                Object[] args = joinPoint.getArgs();
                //调用要记录日志的方法
                res =  joinPoint.proceed(args);
                //获取方法执行时长
                time = System.currentTimeMillis() - time;
                return res;
            } finally {
                try {
                    //方法执行完成后增加日志
                    addOperationLog(joinPoint,res,time);
                }catch (Exception e){
                    System.out.println("LogAspect 操作失败:" + e.getMessage());
                    e.printStackTrace();
                }
            }
        }
    
        private void addOperationLog(JoinPoint joinPoint, Object res, long time){
            //获取当前登录的用户
            UserInfo userInfo = SecurityUserUtil.getThisUserInfo();
            //获取方法的签名,用来获取加在方法上的注解
            MethodSignature signature = (MethodSignature)joinPoint.getSignature();
            //创建日志对象
            OperationLog operationLog = new OperationLog();
            operationLog.setRunTime(time);
            operationLog.setLogId(UUID.randomUUID().toString());
            operationLog.setStartTime(new Date());
            operationLog.setUserName(userInfo.getUserName());
            operationLog.setUserCode(userInfo.getUserCode());
            //获取加在方法上的注解
            OperationLogDetail annotation = signature.getMethod().getAnnotation(OperationLogDetail.class);
            if(annotation != null){
                operationLog.setLogLevel(annotation.logLevel());
                operationLog.setOperationDetail(getDetail(((MethodSignature)joinPoint.getSignature()).getParameterNames(),joinPoint.getArgs(),annotation));
                operationLog.setOperationType(annotation.operationType().getValue());
                operationLog.setOperationUnit(annotation.operationUnit().getValue());
                operationLog.setOperationModule(annotation.operationModule().getValue());
            }
            //保存日志
            operationLogService.insertSystemLog(operationLog);
        }
    
    
        /**
         * 对占位符处理
         * @param argNames 方法参数名称数组
         * @param args 方法参数数组
         * @param annotation 注解信息
         * @return 返回处理后的描述
         */
        private String getDetail(String[] argNames, Object[] args, OperationLogDetail annotation){
    
            Map<Object, Object> map = new HashMap<>(4);
            for(int i = 0;i < argNames.length;i++){
                map.put(argNames[i],args[i]);
            }
            //获取详情信息
            String detail = annotation.operationDetail();
            try {
                //遍历传入方法的参数
                for (Map.Entry<Object, Object> entry : map.entrySet()) {
                    Object k = entry.getKey();
                    Object v = entry.getValue();
                    //request和response不可序列化,XSSFWorkbook也不可序列化
                    if(!(v instanceof HttpServletRequest) && !(v instanceof HttpServletResponse) && !(v instanceof XSSFWorkbook)){
                        if(v instanceof JSONObject){
                            //处理JSONObject格式的参数
                            JSONObject jsonObject = (JSONObject) v;
                            for (String jk : jsonObject.keySet()) {
                                detail = detail.replace("{{" + jk + "}}", jsonObject.get(jk)!=null?jsonObject.get(jk).toString():"");
                            }
                        }else{
                            detail = detail.replace("{{" + k + "}}", JSON.toJSONString(v));
                        }
                    }else if(v instanceof HttpServletRequest){
                        //处理HttpServletRequest
                        JSONObject jsonObject = CommonUtil.request2Json((HttpServletRequest) v);
                        for (String jk : jsonObject.keySet()) {
                            detail = detail.replace("{{" + jk + "}}", jsonObject.get(jk)!=null?jsonObject.get(jk).toString():"");
                        }
                    }
                }
            }catch (Exception e){
                e.printStackTrace();
            }
            return detail;
        }
    
    }

    5、创建Service与Mapper

    public interface OperationLogService {
        //插入
        void insertSystemLog(OperationLog operationLog);
    }
    --------------------------------------------------
    @Service
    public class OperationLogServiceImpl implements OperationLogService {
    
        @Autowired
        private OperationLogMapper operationLogMapper;
        
        @Override
        public void insertSystemLog(OperationLog operationLog) {
            operationLogMapper.insertSystemLog(operationLog);
        }
    }
    --------------------------------------------------
    @Mapper
    @Repository
    public interface OperationLogMapper {
        void insertSystemLog(OperationLog operationLog);
    }
    ---------------------------------------------------
    <?xml version="1.0" encoding="UTF-8"?>
    <!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
            "http://mybatis.org/dtd/mybatis-3-mapper.dtd" >
    <mapper namespace="com.XXX.XXX.dao.OperationLogMapper">
        <insert id="insertSystemLog" parameterType="com.XXX.XXX.domin.OperationLog">
            insert into SYSTEM_OPERATION_LOG(
            LOG_ID,
            USER_CODE,
            USER_NAME,
            START_TIME,
            RUN_TIME,
            OPERATION_UNIT,
            OPERATION_TYPE,
            OPERATION_DETAIL,
            LOG_LEVEL,
            OPERATION_MODULE
            )
            values(
            #{logId},
            #{userCode},
            #{userName},
            #{startTime},
            #{runTime},
            #{operationUnit},
            #{operationType},
            #{operationDetail},
            #{logLevel},
            #{operationModule}
            )
        </insert>
    </mapper>

    使用

    在需要记录日志的方法上添加创建的注解

    @OperationLogDetail(
        operationDetail = "{{userCode}}",   //该占位符将在创建日志对象时扫描参数列表获取
        operationType = OperationType.QUERY,
        operationUnit = OperationUnit.USER_INFO,
        operationModule = OperationModule.USER)
    @PostMapping("/getUserInfo")
    public JSONObject getUserInfo(@RequestBody JSONObject jsonObject){
        return userInfoService.getUserInfo(jsonObject);
    }
  • 相关阅读:
    2018年年终总结
    oracle存储过程 关于update的动态SQL-工作心得
    编译+远程调试spark
    记录一次 hadoop yarn resourceManager无故切换的故障
    hadoop2.8.4 版本yarn RM fairScheduler调度性能优化的若干次尝试
    记录一次hadoop2.8.4版本RM接入zk ha问题
    Ranger-hdfs插件压测
    ranger-hdfs 插件组权限测试
    Nginx基本参数调优
    Nginx安装配置
  • 原文地址:https://www.cnblogs.com/MrZhaoyx/p/12902190.html
Copyright © 2011-2022 走看看