Spring AOP学习笔记

AOP

AOP(Aspect Oriented Programming,面向切面编程)是一种编程范式,与OOP互为补充,初衷是为了解决代码重复问题以及将功能性代码与非功能性代码分离。使用AOP可以集中处理某一关注点,方便地添加/删除关注点。AOP的侵入性也很小。

AOP的应用场景

  • 权限控制
  • 缓存控制
  • 事务控制
  • 审计日志
  • 性能监控
  • 分布式追踪
  • 异常处理

AOP实例

以下模拟新增商品操作,该操作的前提是当前用户为"admin"。

模拟用户切换
public class CurrentUserHolder {

    private static final ThreadLocal<String> HOLDER = new ThreadLocal<>();

    public static String get() {
        return HOLDER.get() == null ? "unknown" : HOLDER.get();
    }

    public static void set(String user) {
        HOLDER.set(user);
    }
}
验证用户身份
@Component
public class AuthUtil {

    /**
     * 验证用户
     */
    public void checkAccess() {
        String user = CurrentUserHolder.get();
        if (!"admin".equals(user)) {
            System.out.println("check access failed.");
            throw new RuntimeException("operation not allow.");
        }
        System.out.println("check access success.");
    }
}
编制切面

@Aspect注解用于标注切面,@Pointcut注解表示切入点,通过切入点表达式标注在哪些类的哪些方法织入,@Before等注解标注了通知(Advice)执行的时机。

@Aspect
@Component
public class AnnotationAspect {

    @Resource
    private AuthUtil authUtil;

    /**
     * 切入点
     */
    @Pointcut("@annotation(com.wch.test.aop.aspectj.AdminOnly)")
    public void adminOnly() {
    }

    @Before("adminOnly()")
    public void check() {
        authUtil.checkAccess();
    }
}
自定义注解

用于标注在需要有"admin"身份才能访问的方法上。

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface AdminOnly {
}
商品操作
@Service
public class ProductService {

    // @Resource
    // private AuthService authService;

    @AdminOnly
    public void insertProduct(Product product) {
        // 不使用aop:在每次需要验证时都手动调用checkAccess()方法
        // authService.checkAccess();
        System.out.println("insert product.");
    }
}

切入点表达式

  • 通配符
    *:匹配任意数量的字符
    +:匹配指定类及其子类
    ..:匹配子包和参数
  • 运算符
    &&:与操作符
    ||:或操作符
    !:非操作符
  • 指示器
    匹配方法:excution()
    匹配注解:@annotation()
    匹配包/类型:within()
    匹配对象:this() / target() / bean()
    匹配参数:args()
使用within()匹配包/类型
@Aspect
@Component
public class WithinAspect {

    /**
     * 匹配类型
     */
    @Pointcut("within(com.wch.test.aop.aspectj.service.impl.AopProductServiceImpl)")
    public void matchType() {
    }

    @Before("matchType()")
    public void before() {
        System.out.println("### before type.");
    }

    /**
     * 匹配包(通过..扩展匹配子包)
     */
    @Pointcut("within(com.wch.test.aop.aspectj.service..*)")
    public void matchPackage() {
    }

    @After("matchPackage()")
    public void after() {
        System.out.println("### after package.");
    }
}
使用this() / target() / bean()匹配对象
@Aspect
@Component
public class ObjectAspect {

    /**
     * 匹配对象
     */
    @Pointcut("this(com.wch.test.aop.aspectj.service.impl.AopProductServiceImpl)")
    public void matchThis() {
    }

    @Before("matchThis()")
    public void before() {
        System.out.println("### before this.");
    }

    /**
     * 匹配实现指定接口的对象
     */
    @Pointcut("target(com.wch.test.aop.aspectj.service.ProductService)")
    public void matchTarget() {
    }

    @After("matchTarget()")
    public void after() {
        System.out.println("### after target.");
    }

    /**
     * 匹配以指定字符串为bean名称结尾的bean
     */
    @Pointcut("bean(*aopProductServiceImpl)")
    public void matchBean() {
    }

    @After("matchBean()")
    public void around() {
        System.out.println("### after bean.");
    }
}
使用args() 匹配指定参数的方法
@Aspect
@Component
public class ArgsAspect {

    /**
     * 匹配含有指定参数的方法
     */
    @Pointcut("args(Long, ..) && within(com.wch.test.aop.aspectj.service..*)")
    public void matchArgs() {
    }

    @Before("matchArgs()")
    public void before() {
        System.out.println("### before args.");
    }

    /**
     * 通过args()获取参数,与before结合可用于检验参数
     *
     * @param id id
     */
    @Before("matchArgs() && args(id)")
    public void getArgsBefore(Long id) {
        System.out.println("### before get args, id = " + id + ".");
    }
}
使用注解匹配
@Aspect
@Component
public class AnnotationAspect {

    @Resource
    private AuthUtil authUtil;

    /**
     * 匹配有指定注解的方法
     */
    @Pointcut("@annotation(com.wch.test.aop.aspectj.AdminOnly)")
    public void adminOnly() {
    }

    @Before("adminOnly()")
    public void before() {
        authUtil.checkAccess();
    }

    /**
     * 匹配有指定注解的类,且注解的RetentionPolicy为CLASS
     */
    @Pointcut("@within(com.wch.test.annotation.custom.Beta)")
    public void within() {
    }

    /**
     * 匹配有指定注解的类,且注解的RetentionPolicy为RUNTIME
     */
    @Pointcut("@target(org.springframework.jmx.export.annotation.ManagedResource)")
    public void target() {
    }

    /**
     * 匹配传入参数含指定注解的方法
     */
    @Pointcut("@args(org.springframework.stereotype.Repository)")
    public void args() {
    }
}
使用execution()匹配指定返回值、方法名、参数的方法
@Aspect
@Component
public class ExecutionAspect {

    /**
     * 匹配指定返回值、方法名、参数的方法
     */
    @Pointcut("execution(public void *..delete*(Long)) throws java.lang.RuntimeException")
    public void matchExecution() {
    }

    @After("matchExecution()")
    public void after() {
        System.out.println("### after execution.");
    }
}
五种Advice注解
  • @Before:前置通知
  • @After:后置通知(方法执行完毕)
  • @AfterReturning:返回通知(成功执行后)
  • @AfterThrowing:异常通知(抛出异常后)
  • @Around:环绕通知
@AfterReturning和@Around的特殊用法
@Pointcut("execution(public String *..AopProduct*..query*(Long))")
    public void matchQueryMethod() {
    }

    /**
     * after returning
     *
     * @param result 获取的指定方法的返回值
     */
    @AfterReturning(value = "matchQueryMethod()", returning = "result")
    public void afterReturning(Object result) {
        System.out.println("### after returning, result: " + result + ".");
    }

    @Pointcut("execution(public boolean *..AopProduct*..update*(Long))")
    public void matchUpdateMethod() {
    }

    /**
     * around
     */
    @Around("matchUpdateMethod()")
    public Object around(ProceedingJoinPoint joinPoint) {
        System.out.println("### before.");
        Object result = null;
        try {
            result = joinPoint.proceed(joinPoint.getArgs());
            System.out.println("### after returning");
        } catch (Throwable e) {
            System.out.println("### after throwing, " + e.getMessage());
        } finally {
            System.out.println("### after");
        }
        return result;
    }
}

Spring AOP实现原理

织入的时机
  • 编译期(AspectJ)
  • 类加载时(AspectJ 5+)
  • 运行时(Spring AOP)
代理模式

调用方通过代理对象来调用目标对象,在代理的过程中,代理对象可以在执行目标对象的过程中额外增加一些操作。

代理模式中代理对象和目标对象都需要实现的接口
public interface Subject {

    void request();
}
目标对象
public class RealSubject implements Subject {

    @Override
    public void request() {
        System.out.println("real subject execute request.");
    }
}
代理对象
public class Proxy implements Subject {

    /**
     * 将目标对象委托给代理对象
     */
    private RealSubject realSubject;

    public Proxy(RealSubject realSubject) {
        this.realSubject = realSubject;
    }

    /**
     * 代理类实现相同方法来对目标对象进行增强
     */
    @Override
    public void request() {
        System.out.println("### before.");
        try {
            realSubject.request();
        } catch (Exception e) {
            System.out.println("ex: " + e.getMessage());
            throw e;
        } finally {
            System.out.println("### after.");
        }
    }
}
调用方
public class Client {

    public static void main(String[] args) {
        Subject subject = new Proxy(new RealSubject());
        subject.request();
    }
}

动态代理

动态代理相对于静态代理的优势在于,当接口中新增方法时,动态代理不必像静态代理需要在目标对象和代理对象中实现该方法,从而提升效率。

JDK动态代理
动态代理类,相当于AOP的aspect
public class JDKProxySubject implements InvocationHandler {

    private RealSubject realSubject;

    public JDKProxySubject(RealSubject realSubject) {
        this.realSubject = realSubject;
    }

    /**
     * 基于反射的动态代理
     *
     * @param proxy  方法反射的代理对象
     * @param method 对象方法
     * @param args   方法参数
     * @return 经过额外操作的方法返回值
     * @throws Throwable Throwable
     */
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("### before.");
        Object result;
        try {
            result = method.invoke(realSubject, args);
        } catch (Exception e) {
            System.out.println("ex: " + e.getMessage());
            throw e;
        } finally {
            System.out.println("### after.");
        }
        return result;
    }
}
调用方
public class JDKProxyClient {

    public static void main(String[] args) {
        Subject subject = (Subject) Proxy.newProxyInstance(JDKProxyClient.class.getClassLoader(),
                new Class[]{Subject.class}, new JDKProxySubject(new RealSubject()));
        subject.request();
    }
}
CGLIB动态代理
基于CGLIB的动态代理
public class CglibMethodInterceptor implements MethodInterceptor {

    /**
     * 通过继承来实现代理
     *
     * @param o           目标对象
     * @param method      目标对象方法
     * @param args        方法参数
     * @param methodProxy 方法代理
     * @return 经过额外操作的方法返回值
     * @throws Throwable Throwable
     */
    @Override
    public Object intercept(Object o, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
        System.out.println("### before.");
        Object result;
        try {
            result = methodProxy.invokeSuper(o, args);
        } catch (Exception e) {
            System.out.println("ex: " + e.getMessage() + ".");
            throw e;
        } finally {
            System.out.println("### after.");
        }
        return result;
    }
}
调用方
public class CglibProxyClient {

    public static void main(String[] args) {
        Enhancer enhancer = new Enhancer();
        enhancer.setSuperclass(RealSubject.class);
        enhancer.setCallback(new CglibMethodInterceptor());
        Subject subject = (Subject) enhancer.create();
        subject.request();
    }
}
JDK动态代理与CGLIB动态代理的区别
  • JDK动态代理只针对有接口的类的方法进行动态代理,无法对private方法代理
  • CGLIB基于继承来实现代理,无法对static、final类代理,无法对private、static方法进行代理
Spring AOP对两种代理的选择方式
  • 如果目标对象实现了接口,则默认采用JDK动态代理
  • 如果目标对象没有实现接口,则采用CGLIB进行动态代理
  • 如果DefaultAopProxyFactory中配置proxyTargetClass = true,则采用CGLIB进行动态代理

AOP的链式调用

责任链模式
链式处理器抽象类
public abstract class ChainHandler {

    /**
     * 继任者
     */
    private ChainHandler successor;

    public ChainHandler getSuccessor() {
        return successor;
    }

    public void setSuccessor(ChainHandler successor) {
        this.successor = successor;
    }

    public void execute() {
        this.handleProcess();
        if (null != successor) {
            // 存在继任者,接替执行execute()方法,实现链式调用
            successor.execute();
        }
    }

    protected abstract void handleProcess();
}
调用方
public class ChainClient {

    static class ChainHandlerA extends ChainHandler {
        @Override
        protected void handleProcess() {
            System.out.println("handle a.");
        }
    }

    static class ChainHandlerB extends ChainHandler {
        @Override
        protected void handleProcess() {
            System.out.println("handle b.");
        }
    }

    static class ChainHandlerC extends ChainHandler {
        @Override
        protected void handleProcess() {
            System.out.println("handle c.");
        }
    }

    public static void main(String[] args) {
        ChainHandlerA handlerA = new ChainHandlerA();
        ChainHandlerB handlerB = new ChainHandlerB();
        ChainHandlerC handlerC = new ChainHandlerC();

        // 将处理器B设置为处理器A的继任者,在A执行后接替执行
        handlerA.setSuccessor(handlerB);
        // 将处理器C设置为处理器B的继任者,在B执行后接替执行
        handlerB.setSuccessor(handlerC);

        // 启动执行A
        handlerA.execute();
    }
}
改进的责任链模式

可以看出以上的责任链模式需要不断去指定处理器的“继任者”,需要进行改良。

链式处理器抽象类
public abstract class ChainHandler {

    /**
     * 链式处理器执行方法
     *
     * @param chain 链式处理器所在处理器链
     */
    public void execute(ProcessChain chain) {
        // 执行处理器本身的逻辑
        this.handleProcess();
        // 通知处理器所在的处理器链继续执行下一个处理器
        chain.proceed();
    }

    protected abstract void handleProcess();
}
执行链
public class ProcessChain {

    /**
     * 链式处理器集合
     */
    private List<ChainHandler> chainHandlers;

    /**
     * 当前执行顺序
     */
    private int index = 0;

    public ProcessChain(List<ChainHandler> chainHandlers) {
        this.chainHandlers = chainHandlers;
    }

    /**
     * 执行链核心处理方法
     */
    public void proceed() {
        if (index < chainHandlers.size()) {
            // 按序处理链式处理器
            chainHandlers.get(index++).execute(this);
        }
    }
}
调用方
public class ChainClient {

    static class ChainHandlerA extends ChainHandler {
        @Override
        protected void handleProcess() {
            System.out.println("handle a.");
        }
    }

    static class ChainHandlerB extends ChainHandler {
        @Override
        protected void handleProcess() {
            System.out.println("handle b.");
        }
    }

    static class ChainHandlerC extends ChainHandler {
        @Override
        protected void handleProcess() {
            System.out.println("handle c.");
        }
    }

    public static void main(String[] args) {
        ChainHandlerA handlerA = new ChainHandlerA();
        ChainHandlerB handlerB = new ChainHandlerB();
        ChainHandlerC handlerC = new ChainHandlerC();

        ProcessChain chain = new ProcessChain(Arrays.asList(handlerA, handlerB, handlerC));
        chain.proceed();
    }
}

AOP应用 —— 实现日志记录功能

将JPA的save(增加或修改)、delete(删除)方法作为切入点,比较增删改操作的前后变更项,将变更记录插入数据库,实现日志功能。

产品订单
@Entity
@Table(name = "product_order")
public class ProductOrder {

    @Id
    @GeneratedValue
    private Long id;

    /**
     * 产品名称
     */
    private String productName;

    /**
     * 订单价格
     */
    private BigDecimal orderPrice;

    /**
     * 下单时间
     */
    @JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss", timezone = "GMT+8")
    private Date createTime;

    /**
     * 修改时间
     */
    @JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss", timezone = "GMT+8")
    private Date updateTime;

    public Long getId() {
        return id;
    }

    public void setId(Long id) {
        this.id = id;
    }

    public String getProductName() {
        return productName;
    }

    public void setProductName(String productName) {
        this.productName = productName;
    }

    public BigDecimal getOrderPrice() {
        return orderPrice;
    }

    public void setOrderPrice(BigDecimal orderPrice) {
        this.orderPrice = orderPrice;
    }

    public Date getCreateTime() {
        return createTime;
    }

    public void setCreateTime(Date createTime) {
        this.createTime = createTime;
    }

    public Date getUpdateTime() {
        return updateTime;
    }

    public void setUpdateTime(Date updateTime) {
        this.updateTime = updateTime;
    }

    @Override
    public String toString() {
        return "ProductOrder{" +
                "id=" + id +
                ", productName='" + productName + '\'' +
                ", orderPrice=" + orderPrice +
                ", createTime=" + createTime +
                ", updateTime=" + updateTime +
                '}';
    }
}
变更项
public class ChangeItem {

    /**
     * 变更属性名
     */
    private String field;

    /**
     * 变更旧值
     */
    private String oldValue;

    /**
     * 变更新值
     */
    private String newValue;

    public String getField() {
        return field;
    }

    public void setField(String field) {
        this.field = field;
    }

    public String getOldValue() {
        return oldValue;
    }

    public void setOldValue(String oldValue) {
        this.oldValue = oldValue;
    }

    public String getNewValue() {
        return newValue;
    }

    public void setNewValue(String newValue) {
        this.newValue = newValue;
    }
}
数据操作记录类型
public enum DataOperationLogType {

    INSERT(1, "新增"),

    UPDATE(2, "修改"),

    DELETE(3, "删除");

    private int code;

    private String info;

    DataOperationLogType(int code, String info) {
        this.code = code;
        this.info = info;
    }

    public int getCode() {
        return code;
    }

    public String getInfo() {
        return info;
    }
}
日志记录
@Entity
@Table(name = "data_operation_log")
public class DataOperationLog {

    @Id
    @GeneratedValue
    private Long id;

    /**
     * 数据操作记录来源
     */
    private String operateMethod;

    /**
     * 被操作的数据记录id
     */
    private Long dataId;

    /**
     * 数据操作时间
     */
    @JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss", timezone = "GMT+8")
    private Date operateTime;

    /**
     * 数据操作类型
     */
    private int dataOperationLogType;

    /**
     * 数据操作变更项集合
     */
    private String changeItems;

    public Long getId() {
        return id;
    }

    public void setId(Long id) {
        this.id = id;
    }

    public String getOperateMethod() {
        return operateMethod;
    }

    public void setOperateMethod(String operateMethod) {
        this.operateMethod = operateMethod;
    }

    public Long getDataId() {
        return dataId;
    }

    public void setDataId(Long dataId) {
        this.dataId = dataId;
    }

    public Date getOperateTime() {
        return operateTime;
    }

    public void setOperateTime(Date operateTime) {
        this.operateTime = operateTime;
    }

    public int getDataOperationLogType() {
        return dataOperationLogType;
    }

    public void setDataOperationLogType(DataOperationLogType dataOperationLogType) {
        this.dataOperationLogType = dataOperationLogType.getCode();
    }

    public String getChangeItems() {
        return changeItems;
    }

    public void setChangeItems(List<ChangeItem> changeItems) {
        this.changeItems = JsonUtils.toJsonString(changeItems);
    }
}
对象属性值变更比较工具
public class FieldValueDiffUtils {

    private static final Logger LOGGER = LoggerFactory.getLogger(FieldValueDiffUtils.class);

    /**
     * 自定义JpaRepository中根据id获取对象的方法为 findOne(需要重写)
     * (继承自org.springframework.data.repository.CrudRepository)
     */
    private static final String ATTAIN_OLD_OBJECT_METHOD = "findOne";

    /**
     * getter方法前缀
     */
    private static final String GETTER_METHOD_PREFIX = "get";

    /**
     * 变更项默认值
     */
    private static final String DEFAULT_ITEM_VALUE = "";

    /**
     * 是否过滤Bean中属性值为null的情况
     */
    private static final boolean IS_FILTER_NULL_BEAN_VALUE = Boolean.TRUE;

    /**
     * 通过反射执行自定义JpaRepository#findOne方法,获取对应id的记录
     *
     * @param target 目标对象
     * @param id     记录id
     * @return 记录
     */
    public static Object getOldObject(Object target, Long id) {
        Object obj = null;
        try {
            Method method = target.getClass().getMethod(ATTAIN_OLD_OBJECT_METHOD, Long.class);
            obj = method.invoke(target, id);
        } catch (NoSuchMethodException | IllegalAccessException | InvocationTargetException e) {
            LOGGER.error("get old object exception: {}", e.getMessage());
        }
        return obj;
    }

    /**
     * 获取指定属性的属性值
     *
     * @param object    目标Bean
     * @param fieldName 属性名称
     * @return 属性值
     */
    private static Object getFieldValue(Object object, String fieldName) {
        Object fieldValue = null;
        if (null != object) {
            Class<?> clazz = object.getClass();
            try {
                // 获取该属性getter方法
                Method method = clazz.getMethod(GETTER_METHOD_PREFIX.
                        concat(fieldName.substring(0, 1).toUpperCase()).
                        concat(fieldName.substring(1)));
                // 通过反射获取属性值
                fieldValue = method.invoke(object);
            } catch (NoSuchMethodException | InvocationTargetException | IllegalAccessException e) {
                LOGGER.error("get field value exception: {}", e.getMessage());
            }
        }
        return fieldValue;
    }

    /**
     * 获取Bean中的属性和值
     *
     * @param object       目标Bean
     * @param isFilterNull {@code true} 过滤值为null的属性,{@code false} 不过滤
     * @return field-value map
     */
    private static Map<String, String> getBeanFieldValueMap(Object object, boolean isFilterNull) {
        Map<String, String> fieldValueMap = new HashMap<>();
        if (null != object) {
            // 获取该Bean的类对象
            Class<?> clazz = object.getClass();
            // 该类的属性集合
            Field[] fields = clazz.getDeclaredFields();
            for (Field field : fields) {
                // 属性名
                String fieldName = field.getName();
                // 属性值
                Object fieldValue = getFieldValue(object, fieldName);
                if (!isFilterNull || null != fieldValue) {
                    fieldValueMap.put(fieldName, JsonUtils.toJsonString(fieldValue));
                }
            }
        }
        return fieldValueMap;
    }

    /**
     * 新增操作,新值整体作为变更项
     *
     * @param object object
     * @return 变更项集合
     */
    public static List<ChangeItem> getInsertChangeItems(Object object) {
        ChangeItem changeItem = new ChangeItem();
        changeItem.setField(DEFAULT_ITEM_VALUE);
        changeItem.setOldValue(DEFAULT_ITEM_VALUE);
        changeItem.setNewValue(JsonUtils.toJsonString(object));
        return Collections.singletonList(changeItem);
    }

    /**
     * 修改操作,比较新原值,获取变更项集合
     *
     * @param oldObject 原值
     * @param newObject 新值
     * @return 变更项集合
     */
    public static List<ChangeItem> getUpdateChangeItems(Object oldObject, Object newObject) {
        List<ChangeItem> changeItems = new ArrayList<>();
        // 获取类对象
        Map<String, String> oldFieldValueMap = getBeanFieldValueMap(oldObject, IS_FILTER_NULL_BEAN_VALUE);
        Map<String, String> newFieldValueMap = getBeanFieldValueMap(newObject, IS_FILTER_NULL_BEAN_VALUE);

        ChangeItem changeItem;
        for (Map.Entry<String, String> newFieldValueEntry : newFieldValueMap.entrySet()) {
            String fieldName = newFieldValueEntry.getKey();
            String newFieldValue = newFieldValueEntry.getValue();
            String oldFieldValue = oldFieldValueMap.get(fieldName);
            if (null == oldFieldValue || !oldFieldValue.equals(newFieldValue)) {
                // 原值不包含此属性或属性值不同,认定为变更项
                changeItem = new ChangeItem();
                changeItem.setField(fieldName);
                changeItem.setOldValue(oldFieldValue);
                changeItem.setNewValue(newFieldValue);
                changeItems.add(changeItem);
            }
        }
        return changeItems;
    }

    /**
     * 删除操作,原值整体作为变更项
     *
     * @param object 原值
     * @return 变更项集合
     */
    public static List<ChangeItem> getDeleteChangeItems(Object object) {
        ChangeItem changeItem = new ChangeItem();
        changeItem.setField(DEFAULT_ITEM_VALUE);
        changeItem.setOldValue(JsonUtils.toJsonString(object));
        changeItem.setNewValue(DEFAULT_ITEM_VALUE);
        return Collections.singletonList(changeItem);
    }
}
日志切面
@Aspect
@Component
public class DataLogAspect {

    @Resource
    private DataOperationLogRepository dataOperationLogRepository;

    private static final Logger LOGGER = LoggerFactory.getLogger(DataLogAspect.class);

    /**
     * ProductOrderRepository#save(ProductOrder productOrder)保存方法名(新增、修改)
     * (继承自org.springframework.data.repository.CrudRepository)
     */
    private static final String SAVE_METHOD_NAME = "save";

    /**
     * ProductOrderRepository#delete(Long id)删除方法名
     * (继承自org.springframework.data.repository.CrudRepository)
     */
    private static final String DELETE_METHOD_NAME = "delete";

    /**
     * 非法ID
     */
    private static final Long INVALID_ID = 0L;

    /**
     * 匹配 ProductOrderRepository#save(ProductOrder productOrder)
     */
    @Pointcut("execution(* com.wch.test.aop.spring.dao.ProductOrderRepository.save(..))")
    public void saveByEntity() {
    }

    /**
     * 匹配 ProductOrderRepository#delete(Long id)
     */
    @Pointcut("execution(* com.wch.test.aop.spring.dao.ProductOrderRepository.delete(..))")
    public void deleteById() {
    }

    /**
     * 针对ProductOrderRepository中两个方法的执行做日志记录
     *
     * @param pjp 连接点
     * @return 目标方法返回值
     */
    @Around("saveByEntity() || deleteById()")
    public Object addOperation(ProceedingJoinPoint pjp) {
        LOGGER.info("enter data log aspect");
        Object returnValue = null;
        try {
            // 获取切入方法所在的目标对象(代理对象)
            Object target = pjp.getTarget();
            // 切入方法名
            String methodName = pjp.getSignature().getName();
            // 切入方法的首个参数
            Object firstArg = pjp.getArgs()[0];
            // 日志记录对象
            DataOperationLog dataOperationLog = new DataOperationLog();
            dataOperationLog.setOperateMethod(methodName);

            /*
             * 组装日志记录
             */
            if (SAVE_METHOD_NAME.equals(methodName)) {
                // 保存方法,参数为ProductOrder对象
                ProductOrder productOrder = (ProductOrder) firstArg;
                Long id = productOrder.getId();
                if (null == id || INVALID_ID.equals(id)) {
                    // 新增
                    List<ChangeItem> insertChangeItems = FieldValueDiffUtils.getInsertChangeItems(productOrder);
                    dataOperationLog.setDataOperationLogType(DataOperationLogType.INSERT);
                    dataOperationLog.setChangeItems(insertChangeItems);
                } else {
                    // 修改
                    Object oldObject = FieldValueDiffUtils.getOldObject(target, id);
                    List<ChangeItem> updateChangeItems = FieldValueDiffUtils.getUpdateChangeItems(oldObject, productOrder);
                    dataOperationLog.setDataOperationLogType(DataOperationLogType.UPDATE);
                    dataOperationLog.setDataId(id);
                    dataOperationLog.setChangeItems(updateChangeItems);
                }
            } else if (DELETE_METHOD_NAME.equals(methodName)) {
                // 删除方法,参数为id
                Long id = (Long) firstArg;
                Object oldObject = FieldValueDiffUtils.getOldObject(target, id);
                List<ChangeItem> deleteChangeItems = FieldValueDiffUtils.getDeleteChangeItems(oldObject);
                dataOperationLog.setDataOperationLogType(DataOperationLogType.DELETE);
                dataOperationLog.setDataId(id);
                dataOperationLog.setChangeItems(deleteChangeItems);
            }

            // 执行目标方法
            returnValue = pjp.proceed(pjp.getArgs());

            if (DataOperationLogType.INSERT.getCode() == dataOperationLog.getDataOperationLogType()) {
                // 新增操作赋新增记录id
                ProductOrder productOrder = (ProductOrder) returnValue;
                dataOperationLog.setDataId(productOrder.getId());
            }

            dataOperationLog.setOperateTime(new Date());
            // 进行日志数据保存
            dataOperationLogRepository.save(dataOperationLog);
            LOGGER.info("log data operation: {}", JsonUtils.toJsonString(dataOperationLog));
        } catch (Throwable e) {
            LOGGER.error("data log operation exception: {}", e.getMessage());
        }
        return returnValue;
    }
}
编写测试
@RunWith(SpringRunner.class)
@SpringBootTest
public class ProductOrderRepositoryTest {

    @Resource
    private ProductOrderRepository productOrderRepository;

    private static final Logger LOGGER = LoggerFactory.getLogger(ProductOrderRepositoryTest.class);

    @Test
    public void insertTest() {
        ProductOrder productOrder = new ProductOrder();
        productOrder.setProductName("Mac Pro");
        productOrder.setOrderPrice(new BigDecimal(13999));
        Date date = new Date();
        productOrder.setCreateTime(date);
        productOrder.setUpdateTime(date);

        ProductOrder order = productOrderRepository.save(productOrder);
        LOGGER.info("order: {}", order);
    }

    @Test
    public void updateTest() {
        ProductOrder productOrder = productOrderRepository.findOne(1L);
        productOrder.setOrderPrice(new BigDecimal(14999));
        productOrder.setUpdateTime(new Date());

        ProductOrder order = productOrderRepository.save(productOrder);
        LOGGER.info("order: {}", order);
    }

    @Test
    public void deleteTest() {
        productOrderRepository.delete(1L);
    }
}
日志记录效果呈现
日志记录
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,423评论 6 491
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,147评论 2 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,019评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,443评论 1 283
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,535评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,798评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,941评论 3 407
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,704评论 0 266
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,152评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,494评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,629评论 1 340
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,295评论 4 329
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,901评论 3 313
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,742评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,978评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,333评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,499评论 2 348