如何知道使用的对象是spring的代理类

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了如何知道使用的对象是spring的代理类相关的知识,希望对你有一定的参考价值。

摘要:在项目中,涉及到同一个类中一个方法调用另外一个方法,并且两个方法的事务不相关,
这里面涉及到一个事务切换的问题,一般的方法没问题,根据通过aop注解在方法上通过加注解标识,
答案是:通过spring aop类里面的AopContext类获取当前类的代理对象,
这样就能切换对应的事务管理器了,具体做法如下:

(1).在applicationContext.xml文件中配置如下:

[html] view plain copy
<!-- 开启暴露Aop代理到ThreadLocal支持 -->
<aop:aspectj-autoproxy expose-proxy="true"/>

(2).在需要切换的地方获取代理对象,再调用对应的方法,如下:

[java] view plain copy
((类名) AopContext.currentProxy()).方法();

(3).注意,这里需要被代理对象使用的方法必须是public类型的方法,不然获取不到代理对象,会报下面的错误:

[java] view plain copy
java.lang.IllegalStateException: Cannot find current proxy: Set 'exposeProxy' property on Advised to 'true' to make it available.
开启暴露AOP代理即可.

因为开启事务和事务回滚,实际这个过程是aop代理帮忙完成的,当调用一个方法时,它会先检查时候有事务,有则开启事务,
当调用本类的方法是,它并没有将其视为proxy调用,而是方法的直接调用,所以也就没有检查该方法是否含有事务这个过程,
那么本地方法调用的事务也就无效了。
参考技术A 在application.yml中有以下配置
spring:
aop:
#auto: true #默认为true,可省略
proxy-target-class: true # 默认为false即JDK动态代理,我们一般要设为true,使用CGLIB代理
这种方式只能将代理模式修改为了CGLIG,但是不能设置暴露cglib代理的目标对象。

对spring创建对象时为何要使用接口

对spring创建对象时为何要使用接口,而使用接口的实现类会报错

接上一篇问题的解答:Spring AOP获取不了增强类(额外方法)和无法通过getBean()获取对象

此问题发生在动态代理时,比如对实现类做增强

对于动态代理分为:JDK动态代理和CGlib动态代理,出现此问题多是这两个代理方式的差异性导致的

Spring默认的类型时JDK动态代理

对实现类对象做增强得到的增强类与实现类是兄弟关系,所以不能用实现类接收增强类对象,只能用接口接收

如果直接使用自然会报错

  UserServiceImpl userService= (UserServiceImpl) context.getBean("UserServicePiont");

但是如果让Spring强制使用CGlib代理

在配置文件aop:config添加proxy-target-class="true则不会报错

<aop:config proxy-target-class="true" >

CGlib代理类和实现类之间是父子关系,自然可以用父类(实现类)去接收子类对象(代理类对象即增强类对象)。

不过应该不会需要这么做,使用接口本来就是解耦的,你直接用实现类接收注入对象岂不是失去了注入的意义。(为什么不直接new一个呢?)

关于动态代理的JDK动态代理和CGlib动态代理还需继续学习

参考:
茅塞顿开,结合网上的对JDK和CGlib理解
spring为何要注入接口,而注入接口的实现类就会报错

以上是关于如何知道使用的对象是spring的代理类的主要内容,如果未能解决你的问题,请参考以下文章

如何用spring生成接口的代理对象并让接口能使用

spring-aop

Spring静态代理与动态代理

Spring中获取被代理的对象

Spring框架中2种生成代理对象的方法

Java设计模式——代理模式