2

我对 AOP 完全陌生。我需要建议来编写正确的切入点。我有一个包含所有服务类的服务包。所有的类都实现了Service接口。这个接口有一个方法save(entity)。每次service.save(entity)方法抛出DataIntegrityViolationException.

这里的方面:

@Component
@Aspect
public class DIVExceptionHandler {
    @AfterThrowing(pointcut = "execution(* myPackage.service.Service.save(*))", throwing = "ex")
        public void handleException(JoinPoint joinPoint, DataIntegrityViolationException ex) {
        //snipped
    }
}

如Spring AOP 文档中所述,我在 CP 中有两个 aspectj jar,并且我已添加<aop:aspectj-autoproxy/>到 Spring 配置中,并且正在使用组件扫描。在测试的日志中,我可以看到方面被检测为 aspetcj 方面:

DEBUG o.s.a.a.a.ReflectiveAspectJAdvisorFactory - Found AspectJ method...

所以我相信这不是配置问题,我的切入点表达是错误的。我也试过

@AfterThrowing(pointcut = "execution(* myPackage.service.*.save(*))", throwing = "ex")

但这也没有用。

那么正确的切入点表达式是什么?

4

1 回答 1

5

这实际上是一个配置问题。

@AfterThrowing(pointcut = "execution(* myPackage.service.Service.save(*))", throwing = "ex")

工作正常。

实际问题DataIntegrityViolationException是仅在代理@Transactional完成事务后才抛出。就我而言,这是在我的建议被调用之后发生的。

解决方案是在事务配置中添加一个订单属性:

<tx:annotation-driven transaction-manager="transactionManager" order="2000"/>

然后@Order向您的方面添加一个小于事务的注释:

@Component
@Order(1500) // must be less than order of <tx:annotation-driven />
@Aspect
public class DIVExceptionHandler {
    @AfterThrowing(pointcut = "execution(* myPackage.service.Service.save(*))", throwing = "ex")
        public void handleException(JoinPoint joinPoint, DataIntegrityViolationException ex) {
        //snipped
    }
}

看:

Spring AOP 排序 - 建议前的交易

于 2013-05-17T07:38:56.537 回答