美文网首页SpringbootSpring
关于cglib或jdk代理的对象无法获取其注解的问题

关于cglib或jdk代理的对象无法获取其注解的问题

作者: 程序员老帮菜 | 来源:发表于2020-05-21 16:48 被阅读0次

背景

同事本想使用applicationContext.getBeansOfType(XXX.class),获取到所有实现类,获取实现类的注解建立子类映射关系,构建策略模式,但是获取到的子类getAnnotation返回为null

同事老工程的代码,如下,可以获取到注解

另外一个工程为什么就获取不到呢???

初步定位问题

  • 老工程获取的子类只是spring管理的实例化的bean
  • 新工程获取到的子类是spring的代理对象
  • spring代理对象不可使用getAnnotation获取注解信息
  • 被代理的类要获取到目标类方可获取注解

梳理问题点

  1. 为什么新工程是代理对象,老工程不是呢
  2. 如何获取被代理对象的目标类
  3. 不获取被代理对象的目标类,有没有方法获取注解

问题解决

为什么新工程是代理对象

  • 新工程使用了aop(比如开启事务、切面),扫描包路径过大,导致所使用的的bean被spring代理
    需要合理规划包路径

如何获取被代理对象的目标类

import org.springframework.aop.framework.AdvisedSupport;
import org.springframework.aop.framework.AopProxy;
import org.springframework.aop.support.AopUtils;

import java.lang.reflect.Field;

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)) {
            proxy = getJdkDynamicProxyTargetObject(proxy);
        } else {
            proxy = getCglibProxyTargetObject(proxy);
        }
        return getTarget(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;
    }
}

不获取被代理对象的目标类,有没有方法获取注解,答案是有的,通过AnnotationUtil方式获取

Order annotation =
 AnnotationUtils.findAnnotation(abstractWordFilter.getClass(), Order.class);
递归解决多重代理问题

相关文章

网友评论

    本文标题:关于cglib或jdk代理的对象无法获取其注解的问题

    本文链接:https://www.haomeiwen.com/subject/wmktahtx.html