反射..获取spring aop代理类 的目标类方法

背景:

在项目开发过程中(注:项目用的SpringMvc),有这么一种需求,系统所有的表格查询都有统一的查询接口,通过通用的Controller类组装查询条件以及调用查询接口返回查询数据。

        但是在实际开发中碰到了这么一个问题,由于项目中用到了springaop(spring用到了cglib和proxy两种方式),所有的数据查询实现类都被包装成SpringAop代理类,遍历类的所有方法获取数据查询方法时,总是获取不到。代码如下所示(正常的想法):


gridService = SpringContextHolder.getBean(service);//通过前端传入的对应业务实现类名获取对应查询类

Method[] methods = gridService.getClass().getMethods();//gridService为数据查询通用接口类

Object param = null;
			
for (Method method : methods) {
				
        Class[] 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代理类( ), 修改配置将 修改成 ,即使用cglib动态代理,情况依然。故分析,只能获取代理类的原始类,即如下代码所示:


 
     /**获取 目标对象
	 * @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



你可能感兴趣的:(java)