zoukankan      html  css  js  c++  java
  • 谈下spring下的 aop日志记录

    在我们开发当中  我们需要对系统用户行为和 系统异常信息有个统一记录  以便后期的 用户行为分析和bug修复   当我们有这个需求时  我们的通常采取方式很多 1.比如我们定义一个规范 开发一个接口  大家在开发的时候 约定形式的 去调用记录日志接口   2 再或者 我们的架构师 或者开发经理 会统一的处理 大家 还是正常的开发 不用去管那些 细节问题  只要按照统一的 规范既可 

    今天 我就来说下 我在项目中的一种 做法,也就是第二种做法:

       大概思想是这样的 : 

                                    1.自定义注解  对我们的Controller Service 进行拦截

                                    2.创建切点类

                                    3.配置spring 对切点进行扫描 实施监控

    下面我上下我的代码:

    @Retention(RetentionPolicy.RUNTIME) // 注解会在class字节码文件中存在,在运行时可以通过反射获取到
    @Target({ElementType.FIELD,ElementType.METHOD})//定义注解的作用目标**作用范围字段、枚举的常量/方法
    @Documented//说明该注解将被包含在javadoc中
    public @interface FieldMeta {

    /**
    * 是否为序列号
    * @return
    */
    boolean id() default false;
    /**
    * 字段名称
    * @return
    */
    String name() default "";
    /**
    * 是否可编辑
    * @return
    */
    boolean editable() default true;
    /**
    * 是否在列表中显示
    * @return
    */
    boolean summary() default true;
    /**
    * 字段描述
    * @return
    */
    String description() default "";
    /**
    * 排序字段
    * @return
    */
    int order() default 0;
    }

         第二步创建 切点类

      

    package com.annotation;

    import com.model.Log;
    import com.model.User;
    import com.service.LogService;
    import com.util.DateUtil;
    import com.util.JSONUtil;
    import com.util.SpringContextHolder;
    import com.util.WebConstants;
    import org.aspectj.lang.JoinPoint;
    import org.aspectj.lang.annotation.*;
    import org.slf4j.Logger;
    import org.slf4j.LoggerFactory;
    import org.springframework.stereotype.Component;
    import org.springframework.web.context.request.RequestContextHolder;
    import org.springframework.web.context.request.ServletRequestAttributes;
    import javax.annotation.Resource;
    import javax.servlet.http.HttpServletRequest;
    import javax.servlet.http.HttpSession;
    import java.lang.reflect.Method;

    /**
    * 切点类
    * @author tiangai
    * @since 2014-08-05 Pm 20:35
    * @version 1.0
    */
    @Aspect
    @Component
    public class SystemLogAspect {
    //注入Service用于把日志保存数据库
    @Resource
    private LogService logService;
    //本地异常日志记录对象
    private static final Logger logger = LoggerFactory.getLogger(SystemLogAspect. class);

    //Service层切点
    @Pointcut("@annotation(com.annotation.SystemServiceLog)")
    public void serviceAspect() {
    }

    //Controller层切点
    @Pointcut("@annotation(com.annotation.SystemControllerLog)")
    public void controllerAspect() {
    }

    /**
    * 前置通知 用于拦截Controller层记录用户的操作
    *
    * @param joinPoint 切点
    */
    @Before("controllerAspect()")
    public void doBefore(JoinPoint joinPoint) {

    HttpServletRequest request = ((ServletRequestAttributes) RequestContextHolder.getRequestAttributes()).getRequest();
    HttpSession session = request.getSession();
    //读取session中的用户
    User user = (User) session.getAttribute(WebConstants.CURRENT_USER);
    //请求的IP
    String ip = request.getRemoteAddr();
    try {
    //*========控制台输出=========*//
    System.out.println("=====前置通知开始=====");
    System.out.println("请求方法:" + (joinPoint.getTarget().getClass().getName() + "." + joinPoint.getSignature().getName() + "()"));
    System.out.println("方法描述:" + getControllerMethodDescription(joinPoint));
    System.out.println("请求人:" + user.getName());
    System.out.println("请求IP:" + ip);
    //*========数据库日志=========*//
    Log log = SpringContextHolder.getBean("logxx");
    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( null);
    log.setCreateBy(user);
    log.setCreateDate(DateUtil.getCurrentDate());
    //保存数据库
    logService.add(log);
    System.out.println("=====前置通知结束=====");
    } catch (Exception e) {
    //记录本地异常日志
    logger.error("==前置通知异常==");
    logger.error("异常信息:{}", e.getMessage());
    }
    }

    /**
    * 异常通知 用于拦截service层记录异常日志
    *
    * @param joinPoint
    * @param e
    */
    @AfterThrowing(pointcut = "serviceAspect()", throwing = "e")
    public void doAfterThrowing(JoinPoint joinPoint, Throwable e) {
    HttpServletRequest request = ((ServletRequestAttributes) RequestContextHolder.getRequestAttributes()).getRequest();
    HttpSession session = request.getSession();
    //读取session中的用户
    User user = (User) session.getAttribute(WebConstants.CURRENT_USER);
    //获取请求ip
    String ip = request.getRemoteAddr();
    //获取用户请求方法的参数并序列化为JSON格式字符串
    String params = "";
    if (joinPoint.getArgs() != null && joinPoint.getArgs().length > 0) {
    for ( int i = 0; i < joinPoint.getArgs().length; i++) {
    params += JSONUtil.toJsonString(joinPoint.getArgs()[i]) + ";";
    }
    }
    try {
    /*========控制台输出=========*/
    System.out.println("=====异常通知开始=====");
    System.out.println("异常代码:" + e.getClass().getName());
    System.out.println("异常信息:" + e.getMessage());
    System.out.println("异常方法:" + (joinPoint.getTarget().getClass().getName() + "." + joinPoint.getSignature().getName() + "()"));
    System.out.println("方法描述:" + getServiceMthodDescription(joinPoint));
    System.out.println("请求人:" + user.getName());
    System.out.println("请求IP:" + ip);
    System.out.println("请求参数:" + params);
    /*==========数据库日志=========*/
    Log log = SpringContextHolder.getBean("logxx");
    log.setDescription(getServiceMthodDescription(joinPoint));
    log.setExceptionCode(e.getClass().getName());
    log.setType("1");
    log.setExceptionDetail(e.getMessage());
    log.setMethod((joinPoint.getTarget().getClass().getName() + "." + joinPoint.getSignature().getName() + "()"));
    log.setParams(params);
    log.setCreateBy(user);
    log.setCreateDate(DateUtil.getCurrentDate());
    log.setRequestIp(ip);
    //保存数据库

    //调用线程保存至数据库
    ThreadPoolUtil.getPool().execute(new SaveSystemLogThread(log,logService));

    System.out.println("=====异常通知结束=====");
    } catch (Exception ex) {
    //记录本地异常日志
    logger.error("==异常通知异常==");
    logger.error("异常信息:{}", ex.getMessage());
    }
    /*==========记录本地异常日志==========*/
    logger.error("异常方法:{}异常代码:{}异常信息:{}参数:{}", joinPoint.getTarget().getClass().getName() + joinPoint.getSignature().getName(), e.getClass().getName(), e.getMessage(), params);

    }


    /**
    * 获取注解中对方法的描述信息 用于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;
    }
    }

    第三步把Controller的代理权交给cglib


    在实例化ApplicationContext的时候需要加上


    Xml代码
    <!-- 启动对@AspectJ注解的支持 -->
    <aop:aspectj-autoproxy/>
    在调用Controller的时候AOP发挥作用所以在SpringMVC的配置文件里加上

    Xml代码
    <!--通知spring使用cglib而不是jdk的来生成代理方法 AOP可以拦截到Controller->
    <aop:aspectj-autoproxy proxy-target-class="true" />

    使用 层次列子

      

    /**
    * 删除用户
    *
    * @param criteria 条件
    * @param id id
    * @param model 模型
    * @return 数据列表
    */
    @RequestMapping(value = "/delete")
    //此处为记录AOP拦截Controller记录用户操作
    @SystemControllerLog(description = "删除用户")
    public String del(Criteria criteria, String id, Model model, HttpSession session) {
    try {
    User user = (User) session.getAttribute(WebConstants.CURRENT_USER);
    if ( null != user) {
    if (user.getId().equals(id)) {
    msg = "您不可以删除自己!";
    criteria = userService.selectByCriteriaPagination(criteria);
    } else {
    //删除数据并查询出数据
    criteria = userService.delete(id, criteria);
    msg = "删除成功!";
    }
    }
    } catch (Exception e) {
    msg = "删除失败!";
    } finally {
    model.addAttribute("msg", msg);
    model.addAttribute("criteria", criteria);
    }
    //跳转列表页
    return "user/list";
    }

    service层次的

    /**
    * 按照分页查询
    * @param criteria
    * @return
    */
    //此处为AOP拦截Service记录异常信息。方法不需要加try-catch
    @SystemServiceLog(description = "查询用户")
    public Criteria<User> selectByCriteriaPagination(Criteria<User> criteria)
    {
    criteria.getList().get(0).getAccount();
    //查询总数
    long total=userMapper.countByCriteria(criteria);
    //设置总数
    criteria.setRowCount(total);
    criteria.setList(userMapper.selectByCriteriaPagination(criteria));
    return criteria;
    }

    在我们的这个 aop日志下的处理 我们又做了 线程方面的处理

      

    package cn.exrick.common.utils;

    import java.util.concurrent.ArrayBlockingQueue;
    import java.util.concurrent.BlockingQueue;
    import java.util.concurrent.ThreadPoolExecutor;
    import java.util.concurrent.TimeUnit;

    /**
    * @author Exrickx
    */
    public class ThreadPoolUtil {

    //线程缓冲队列
    private static BlockingQueue<Runnable> bqueue = new ArrayBlockingQueue<Runnable>(100);
    // 核心线程数,会一直存活,即使没有任务,线程池也会维护线程的最少数量
    private static final int SIZE_CORE_POOL = 5;
    // 线程池维护线程的最大数量
    private static final int SIZE_MAX_POOL = 10;
    // 线程池维护线程所允许的空闲时间
    private static final long ALIVE_TIME = 2000;


    private static ThreadPoolExecutor pool = new ThreadPoolExecutor(SIZE_CORE_POOL, SIZE_MAX_POOL, ALIVE_TIME, TimeUnit.MILLISECONDS, bqueue, new ThreadPoolExecutor.CallerRunsPolicy());

    static {

    pool.prestartAllCoreThreads();
    }

    public static ThreadPoolExecutor getPool() {
    return pool;
    }

    public static void main(String[] args) {
    System.out.println(pool.getPoolSize());
    }
    }

     

     

     如有疑问 请加QQ 344436738 探讨   

  • 相关阅读:
    A.02.01—功能定义—一般定义
    A.02.00—功能定义与唤醒—起始
    A.01.12—模块的输出—通讯(CAN&LIN)
    A.01.11—模块的输出—输出复用和可配
    A.01.10—模块的输出—PWM高端输出
    A.01.09—模块的输出—PWM低端输出
    redis命令
    memcached命令
    kafka命令
    nginx命令
  • 原文地址:https://www.cnblogs.com/AnKangwenqiang/p/8526888.html
Copyright © 2011-2022 走看看