1. 程式人生 > >反射..獲取spring aop代理類 的目標類方法

反射..獲取spring aop代理類 的目標類方法

背景:

在專案開發過程中(注:專案用的SpringMvc),有這麼一種需求,系統所有的表格查詢都有統一的查詢介面,通過通用的Controller類組裝查詢條件以及呼叫查詢介面返回查詢資料。

        但是在實際開發中碰到了這麼一個問題,由於專案中用到了springaop(spring用到了cglib和proxy兩種方式),所有的資料查詢實現類都被包裝成SpringAop代理類,遍歷類的所有方法獲取資料查詢方法時,總是獲取不到。程式碼如下所示(正常的想法):

gridService = SpringContextHolder.getBean(service);//通過前端傳入的對應業務實現類名獲取對應查詢類

<strong><span style="color:#ff0000;">Method[] methods = gridService.getClass().getMethods();//gridService為資料查詢通用介面類</span></strong>

Object param = null;
			
for (Method method : methods) {
				
        Class<? extends Object>[] clazz = method.getParameterTypes();
				
	    if (method.getName().equals("queryItems") && clazz.length == 1) {
					
		if (clazz[0].getName().equals(Object.class.getName())) {
			continue;
		}
					
		param = org.springframework.beans.BeanUtils.instantiateClass(clazz[0]);
					
		break;
        }
}
List datas = gridService.queryItems(param);//datas為查詢後的結果

測試結果,param一直為空,也就是說找不到對應查詢的方法。是怎麼回事呢? 後來跟蹤原始碼檢視,發現gridService為springaop包裝的aop代理類(<aop:config>), 修改配置將<aop:config>修改成<aop:config proxy-target-class="true">,即使用cglib動態代理,情況依然。故分析,只能獲取代理類的原始類,即如下程式碼所示:
 <pre name="code" class="java">     /**獲取 目標物件
	 * @param proxy 代理物件
	 */
	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);
		}
	}

	/** CGLIBProxy 返回目標類 **/
	private static Object getCglibProxyTargetObject(Object proxy) throws Exception {
		
		Object dynamicAdvisedInterceptor = ReflectionUtils.getFieldValue(proxy, "CGLIB$CALLBACK_0");

		Object target = ((AdvisedSupport) ReflectionUtils.getFieldValue(dynamicAdvisedInterceptor, "advised")).getTargetSource().getTarget();

		return target;
	}

	/** JDKProxy 返回目標類 **/
	private static Object getJdkDynamicProxyTargetObject(Object proxy) throws Exception {
		
		InvocationHandler invocationHandler = Proxy.getInvocationHandler(proxy);

		Object target = ((AdvisedSupport) ReflectionUtils.getFieldValue(invocationHandler, "advised")).getTargetSource().getTarget();

		return target;
	}

Method[] methods = gridService.getClass().getMethods(); 改成

  Method[] methods =getTarget(gridService).getClass().getMethods()

就可以查出來資料了

By: http://jianglanlan.iteye.com/blog/1855223