参考文档:Spring 获取代理对象中被代理的目标对象

1. 代理对象

为什么对象会被代理?

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

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

  1. 编写单元测试时向其中注入 mock 对象。(旧版本Spring 的 ReflectionTestUtils 中不支持向代理对象注入。按照这里的说法,spring-test版本升级到4.3.13.RELEASE 才会支持。)
  2. 单元测试中,我们注入 bean 时,注入的是被代理对象,我们无法获取被代理对象的 private 方法,因为代理对象只代理了 public 方法,自身并没有 private 方法,我们需要获取代理对象的目标对象来做反射。
  3. 其他奇怪的需求。

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 方法。