JAVA同一个类内方法调用AOP解决办法

1. 问题现象

    @Around("execution(* com.unicom.hl.service.TowerPayService.processInsert(..))")
    public Object around(ProceedingJoinPoint pjp) throws Throwable {
    	...
    }

可以看到很明确的对一个方法进行了切面定义。但是在执行的时候却没有走切面的逻辑。
业务代码里的调用如下

	// 上层调用这个方法,这个方法在调用被切面的方法
	public Object insertPay(List<PayInfo> list) {
		...
		processInsert_db(province, list);  // 在方法内部调用定义的切面方法
		...
	}

	// 要做切面的方法
    @TargetDataSource
    public List<PayLineResponse> processInsert_db(String province, List<PayInfo> list) {
    	...
    }

2. 原因分析

问题就在于调用切面方法processInsert_db的方法是同一个类内部的。如果是其他类直接调用processInsert_db那么会走切面逻辑。
因为AOP就是对原有的业务逻辑做一次增强,在原有代码的基础上在生成class文件的时候或者装在class是在JVM内生成一个代理代理对象,代理对象被@Autowired注入到其他类的引用里,所以当其他类调用时实际上使用的是代理对象,所以会触发切面逻辑。 但如果是类内部调用,使用的是对象本身,而不是代理对象,所以不走切面逻辑。

3. 解决方案

方案一:通过代码解决

在spring boot的启动类上增加注解@EnableAspectJAutoProxy(exposeProxy=true)
在类内部调用的时候使用如下方式获取代理对象,所以根源还是要调用代理对象的方法,而不是原对象本身。
把上边的调用改成如下方式

	public Object insertPay(List<PayInfo> list) {
		...
		PayService service = AopContext.currentProxy() != null ? (PayService)AopContext.currentProxy() : this;
        List<PayLineResponse> resultResponses = service.processInsert_db(province, subList);
		...
	}
方案二:不使用切面

不使用切面,直接把逻辑写在代码里。或者其他变通的方式实现。或者让上一层代码调用这个代理对象的方法

  • 2
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
Java 中,使用 @Transactional 注解可以实现事务管理。但是在某些情况下,@Transactional 注解可能不会生效,比如注解在事务的内部方法上。 这是因为 Spring AOP 默认只拦截通过 Spring 容器管理的 Bean 的方法调用,而事务通常是使用自调用的方式,因此不会被 Spring AOP 拦截。 解决这个问题的方法是使用 AspectJ。AspectJ 是一种基于语言的 AOP 实现方式,可以在编译期或者运行期织入切面。 具体实现如下: 1. 添加 AspectJ 依赖 ```xml <dependency> <groupId>org.springframework</groupId> <artifactId>spring-aspects</artifactId> <version>${spring.version}</version> </dependency> ``` 2. 配置 Spring,启用 AspectJ 自动代理 ```xml <bean class="org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor" /> <bean class="org.springframework.context.annotation.CommonAnnotationBeanPostProcessor" /> <aop:aspectj-autoproxy /> ``` 3. 在事务上添加 @Aspect 注解,并实现一个切面 ```java @Aspect @Component public class TransactionAspect { @Autowired private PlatformTransactionManager transactionManager; @Pointcut("@annotation(org.springframework.transaction.annotation.Transactional)") public void transactionalMethod() {} @Around("transactionalMethod()") public Object doTransaction(ProceedingJoinPoint joinPoint) throws Throwable { TransactionStatus status = transactionManager.getTransaction(new DefaultTransactionDefinition()); Object result; try { result = joinPoint.proceed(); transactionManager.commit(status); } catch (Exception e) { transactionManager.rollback(status); throw e; } return result; } } ``` 这个切面会拦截所有被 @Transactional 注解标记的方法,并使用事务管理器执行事务。 以上就是解决注解@Transactional事务类内调用不生效问题的方法

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值