zoukankan      html  css  js  c++  java
  • spring-aop

    package com.cy.pj.common.aspect;
    import java.lang.reflect.Method;
    import java.util.Arrays;
    import java.util.Date;

    import org.apache.shiro.SecurityUtils;
    import org.aspectj.lang.ProceedingJoinPoint;
    import org.aspectj.lang.Signature;
    import org.aspectj.lang.annotation.Around;
    import org.aspectj.lang.annotation.Aspect;
    import org.aspectj.lang.reflect.MethodSignature;
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.stereotype.Service;

    import com.cy.pj.common.annotation.RequiredLog;
    import com.cy.pj.common.util.IPUtils;
    import com.cy.pj.sys.dao.SysLogDao;
    import com.cy.pj.sys.entity.SysLog;
    import com.cy.pj.sys.entity.SysUser;
    /**
    * @Aspect 注解修饰的类通常认为一个切面对象类型
    * 切面对象是对扩展业务的封装,它通常会在内部声明
    * 如下几个部分.
    * 1)实现扩展业务的方法(一般会称为通知-advice)
    * 2)切入扩展业务的点(一般会称为切入点-PointCut)
    */
    //@Order(1)
    @Aspect
    @Service
    public class SysLogAspect {//日志切面
    /**
    * @Around注解修饰方法为一个环绕通知,其目的
    * 是在目标业务方法执行之前和之后都可以进行
    * 扩展业务的处理
    * 其中:
    * 1)bean(sysUserServiceImpl) 为切入点表达式,
    * 表示sysUserServiceImpl对象中所有业务方法执行
    * 时都会执行@Around注解修饰的方法
    * @param jp 连接点(封装了要执行的目标方法信息)
    * @return
    * @throws Throwable
    */
    //@Around("bean(sysUserServiceImpl)")
    //@Around("bean(*ServiceImpl)")
    //@annotation()为细粒度的切入点表达式定义方式
    @Around("@annotation(com.cy.pj.common.annotation.RequiredLog)")
    public Object aroundMethod(ProceedingJoinPoint jp)
    throws Throwable{
    System.out.println("LogAspect:开始记录日志");
    //1.目标业务执行之前的记录
    long t1=System.currentTimeMillis();
    //2.执行目标业务(底层通过反射执行目标方法)
    Object result=jp.proceed();
    //3.目标业务执行之后的记录
    long t2=System.currentTimeMillis();
    System.out.println("目标业务执行时长:"+(t2-t1));
    saveObject(jp,(t2-t1));
    //4.返回目标业务的执行结果
    return result;
    }
    @Autowired
    private SysLogDao sysLogDao;
    private void saveObject(ProceedingJoinPoint jp,long time)throws Exception {
    //1.获取要保存的日志信息
    //1.1获取登陆用户(没问题)
    SysUser user=(SysUser)SecurityUtils.getSubject().getPrincipal();
    //1.2获取方法签名(此对象封装了我们要执行的目标方法信息)
    Signature s=jp.getSignature();
    System.out.println(s.getClass().getName());//MethodSignature
    MethodSignature ms=(MethodSignature)s;
    //1.2.1获取目标对象(要执行的业务层对象)
    Class<?> targetClass=jp.getTarget().getClass();
    //1.2.2基于目标业务对象获取要执行的目标方法
    //?思考(为什么要获取此方法呢)
    Method targetMethod=targetClass.getDeclaredMethod(
    ms.getName(),
    ms.getParameterTypes());
    //1.2.3获取方法上定义的注解内容(定义的操作名)
    RequiredLog requiredLog=
    targetMethod.getDeclaredAnnotation(RequiredLog.class);
    String operation=requiredLog.value();
    //1.2.4获取目标对象方法的全称(类全名+方法名)
    String targetClassName=targetClass.getName();
    String targetMethodName=targetClassName+"."+targetMethod.getName();
    //1.3获取方法执行时的实际参数
    String params=Arrays.toString(jp.getArgs());
    //2.封装日志信息
    SysLog log=new SysLog();
    log.setUsername(user.getUsername());
    log.setIp(IPUtils.getIpAddr());
    log.setOperation(operation);
    log.setMethod(targetMethodName);
    log.setParams(params);
    log.setTime(time);
    log.setCreatedTime(new Date());
    //3.将日志信息写入到数据库
    sysLogDao.insertObject(log);
    }

    }

  • 相关阅读:
    面向对象进阶
    初识面向对象
    模块和包
    day 17递归函数
    pip命令无法使用
    Python中的iteritems()和items()
    C# 截取字符串
    Python连接Mysql数据库
    【PYTHON】 Missing parentheses in call to 'print'
    javaScript中with函数用法实例分析
  • 原文地址:https://www.cnblogs.com/jishumonkey/p/12877346.html
Copyright © 2011-2022 走看看