反射..獲取spring aop代理類 的目標類方法
阿新 • • 發佈:2019-01-31
背景:
在專案開發過程中(注:專案用的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