在spring中獲取代理物件代理的目標物件工具類

jasin-i技術視界發表於2020-10-17

 

問題描述

我現在遇到個棘手的問題,要通過spring託管的service類儲存物件,這個類是通過反射拿到的,經過實驗發現這個類只能反射取得sservice實現了介面的方法,而extends類的方法一律不出現,debug後發現這個servie例項被spring替換成jdkdynmicproxy類,而不是原始物件了,,它裡面只有service繼承的介面方法,而沒有extends 過的super class方法,怎麼呼叫原生物件的方法!!!!!

用託管的spring service類呼叫getClass().getName()方法,發現輸出都是$proxy43這類東西!!

通過此種方式獲取目標物件是不可靠的,或者說任何獲取目標物件的方式都是不可靠的,因為TargetSource,TargetSource中存放了目標物件,但TargetSource有很多種實現,預設我們使用的是SingletonTargetSource ,但還有其他的比如ThreadLocalTargetSource、CommonsPoolTargetSource 等等。

這也是為什麼spring沒有提供獲取目標物件的API。

解決方法

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

}  

相關文章