Spring 获取代理对象中被代理的目标对象


#Spring#


什么对象会被代理?

  1. 被 AOP 加入了切面的对象。
  2. 使用了 @Transactional 的对象。
  3. 等等。

为什么要获取被代理的目标对象?

  1. 编写单元测试时向其中注入 mock 对象。(旧版本Spring 的 ReflectionTestUtils 中不支持向代理对象注入。按照这里的说法,spring-test版本升级到4.3.13.RELEASE 才会支持。)
  2. 其他奇怪的需求。

如何获取?

方案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;  
    }  
      
}

上面这段代码,网络上很多地方能搜到。例如:

方案2

在新版本的 spring-test (比如 spring-test 5.0.6.RELEASE)中可以用 AopTestUtils.getUltimateTargetObject 方法。


( 本文完 )