spring框架aop用注解形式注入Aspect切面无效的问题解决

时间:2023-03-10 06:38:36
spring框架aop用注解形式注入Aspect切面无效的问题解决

由于到最后我的项目还是有个邪门的错没解决,所以先把文章大概内容告知:

1.spring框架aop注解扫描默认是关闭的,得手动开启。

2.关于Con't call commit when autocommit=true报错信息的解决。

3.最后是有一个由bean.xml配置切面(运行结果没问题)转换成注解配置切面(结果:执行方法顺序错误,而且注解的位置我也没注错)的问题。希望有大佬路过可以帮我看一看。

如果前两条可能帮不上你,又对我遇到的错误没兴趣,又在着急解决自己遇到的问题的话,可以继续去搜索其他解决办法了。

正文:

学习spring框架并练习转账业务的代码编写

为转账业务已经写好事务控制类,保证每一个业务用同一个connection。并且已经在bean.xml中用<aop:config>标签配置好切面,并且能成功走切面。

而我用注解的形式代替用bean.xml<aop:config>标签形式完成切面配置后发现不走切面,就是我写的事务控制类即使加了注解也不走事务控制类里的方法了。

我怀疑是不是漏掉了什么,于是上网查找annotation配置步骤,发现果然缺了一步:开启aop注解扫描方式。

aop注解扫描默认是false,得在bean.xml中加上这段代码:

<!--开启aop注解方式,默认为false-->
<aop:aspectj-autoproxy ></aop:aspectj-autoproxy>

运行后果然走切面了,但是紧接着报了一个错

spring框架aop用注解形式注入Aspect切面无效的问题解决

查了一下报错原因:

Can't call commit when autocommit=true表示的意思就是当自动属性值为true是无法call commit(mysql默认都是自动提交)

解决此问题只需要将自动提交改成手动提交即可,也就是令autocommit=false。

我找到提示得错误位置的代码

spring框架aop用注解形式注入Aspect切面无效的问题解决

意思是提交之前得将自动提交改成手动提交,于是我将代码改成:

spring框架aop用注解形式注入Aspect切面无效的问题解决

运行之后成功了。但是我的想法是转账业务得方法是切点,开启事务是前置通知,提交事务时后置通知。

并且已经在前置通知里关闭自动提交。

spring框架aop用注解形式注入Aspect切面无效的问题解决

按理说用的是同一条连接,我应该不用在提交之前重新再关一遍自动提交。后来在运行结果出发现了问题。

spring框架aop用注解形式注入Aspect切面无效的问题解决

业务处理执行完毕之后并没有提交,而是释放了资源,释放资源之后才进行了提交操作。导致还是用了两个连接(connection),而且有一个挺搞笑的事:

第一个连接做的事是:开始事务:方法是关闭了自动提交。执行业务:感觉上对数据库一顿操作,但是没提交,也不会自动提交。最后释放资源了:将连接关闭了。

第二两连接做的事是:提交事务。他也不知道要提交啥,最后还没被关闭,他估计现在还在想呢,我刚刚到底提交了啥?

这一波操作,代码通了,也没报错。你感觉你成了,其实数据库没变。一切跟没做一样。

总而言之,我好像知道哪里出错了,应该是切面里的方法执行顺序错了,我第一感觉应该是我马虎将方法上注解加错了。去检查一番发现好像并没错

切点表达式是对的,因为从运行结果上看,已经找到切点了。前置通知用Before,后置通知用AfterReturning,最终通知用After。这些都没错。

我彻底懵了。处世未深的我瑟瑟发抖。用bean.xml配置没有问题,改成注解配置怎么会出现顺序的错误呢?

可能我学的时候忽略了什么知识,或者有地方马虎了我还没找出来。希望大佬们帮忙看看。以下是可能出现问题的代码(只进行改动的代码):

bean.xml(用这种方式运行是没错的):

<!--Aspect已经用注解形式在切面类TranscationManager.java中配置完成-->
<!--开启aop注解方式,默认为false-->
<!--<aop:aspectj-autoproxy ></aop:aspectj-autoproxy>-->
<aop:config>
<aop:pointcut id="pt1" expression="execution(* ccut.wulang.service.impl.*.*(..))"/>
<aop:aspect id="txValue" ref="transcationManager">
<aop:before method="beginTranscation" pointcut-ref="pt1"></aop:before>
<aop:after-returning method="commit" pointcut-ref="pt1"></aop:after-returning>
<aop:after-throwing method="rollback" pointcut-ref="pt1"></aop:after-throwing>
<aop:after method="release" pointcut-ref="pt1"></aop:after>
<!--<aop:around method="around" pointcut-ref="pt1"></aop:around>-->
</aop:aspect>
</aop:config>

TranscationManager.java切面类:

@Component
@Aspect
public class TranscationManager {
@Autowired
ConnectionUtil connectionUtil;
@Pointcut(value = "execution(* ccut.wulang.service.impl.*.*(..))")
private void pt1(){

}
@Before("pt1()")
public void beginTranscation(){
System.out.println("开启事务....");
try {
connectionUtil.getThreadConnect().setAutoCommit(false);
} catch (SQLException throwables) {
throwables.printStackTrace();
}
}

@AfterReturning("pt1()")
public void commit(){
System.out.println("提交事务....");
try {
connectionUtil.getThreadConnect().commit();
} catch (SQLException throwables) {
throwables.printStackTrace();
}
}
@AfterThrowing("pt1()")
public void rollback(){
System.out.println("回滚事务....");
try {
connectionUtil.getThreadConnect().rollback();
} catch (SQLException throwables) {
throwables.printStackTrace();
}
}
@After("pt1()")
public void release(){
System.out.println("释放资源....");
try {
connectionUtil.getThreadConnect().close();
connectionUtil.close();
} catch (SQLException throwables) {
throwables.printStackTrace();
}
}
}