1. 代理对象
为什么对象会被代理?
- 被 AOP 加入了切面的对象。
- 使用了 @Transactional 的对象。
- 等等。
2. 为什么要获取被代理的目标对象
- 编写单元测试时向其中注入 mock 对象。(旧版本Spring 的 ReflectionTestUtils 中不支持向代理对象注入。按照这里的说法,spring-test版本升级到4.3.13.RELEASE 才会支持。)
- 单元测试中,我们注入 bean 时,注入的是被代理对象,我们无法获取被代理对象的 private 方法,因为代理对象只代理了 public 方法,自身并没有 private 方法,我们需要获取代理对象的目标对象来做反射。
- 其他奇怪的需求。
3. 如何获取
3.1 方案一
import java.lang.reflect.Field;
import org.springframework.aop.framework.AdvisedSupport;
import org.springframework.aop.framework.AopProxy;
import org.springframework.aop.support.AopUtils;
public class AopTargetUtils {
/**
* 获取 目标对象
* @param proxy 代理对象
* @return
* @throws Exception
*/
public static Object getTarget(Object proxy) throws Exception {
if(!AopUtils.isAopProxy(proxy)) {
return proxy; //不是代理对象
}
if(AopUtils.isJdkDynamicProxy(proxy)) {
return getJdkDynamicProxyTargetObject(proxy);
} else { //cglib
return getCglibProxyTargetObject(proxy);
}
}
private static Object getCglibProxyTargetObject(Object proxy) throws Exception {
Field h = proxy.getClass().getDeclaredField("CGLIB$CALLBACK_0");
h.setAccessible(true);
Object dynamicAdvisedInterceptor = h.get(proxy);
Field advised = dynamicAdvisedInterceptor.getClass().getDeclaredField("advised");
advised.setAccessible(true);
Object target = ((AdvisedSupport)advised.get(dynamicAdvisedInterceptor)).getTargetSource().getTarget();
return target;
}
private static Object getJdkDynamicProxyTargetObject(Object proxy) throws Exception {
Field h = proxy.getClass().getSuperclass().getDeclaredField("h");
h.setAccessible(true);
AopProxy aopProxy = (AopProxy) h.get(proxy);
Field advised = aopProxy.getClass().getDeclaredField("advised");
advised.setAccessible(true);
Object target = ((AdvisedSupport)advised.get(aopProxy)).getTargetSource().getTarget();
return target;
}
}
3.2 方案二
在新版本的 spring-test (比如 spring-test 5.0.6.RELEASE)中可以用 AopTestUtils.getUltimateTargetObject 方法。