getAnnotation(Class.class) 为空问题

您所在的位置:网站首页 bean值为null getAnnotation(Class.class) 为空问题

getAnnotation(Class.class) 为空问题

#getAnnotation(Class.class) 为空问题| 来源: 网络整理| 查看: 265

    今天写maven插件,需要实现扫描自定义注解,但是死活通过Class.getAnnotation(Annotation.class)拿不到注解,一直返回空,网上都说是注解没有加上@Retention(RetentionPolicy.RUNTIME),但其实我是有的,通过在被扫描类里写main方法,一样的代码是可以拿到注解的。

    通过调试,发现getAnnotation方法内部其实把Class对象的所有Annotation都放到了一个Map中,最后通过传入了注解class对象作为Key去找是否存在。最后发现,main方法中的ket存在,而插件中的key不存在。说明两个Annotation.class并不是同一个对象。

    那么为什么插件中就不是一个对象呢?

    插件中的需求是根据目标输出jar读取扫描jar中类是否包含指定注解,但是插件本身classpath并没有目标jar,这个时候就要自己使用classload加载jar了,实际上classLoad加载了目标class和注解类,代码贴上:

private void ResolveClass(String className) { try { Class clazz = classLoader.loadClass(className); Class aClass = classLoader.loadClass(MyApi.class.getName()); System.out.println(clazz.getName()); MyApi annotation = clazz.getAnnotation(MyApi.class); if(annotation != null) { System.out.println(((MyApi) annotation).name() + ((MyApi) annotation).desc()); } } catch (ClassNotFoundException e) { e.printStackTrace(); } catch (NoSuchMethodException e) { e.printStackTrace(); } }

    每个类的Class对象的创建跟随classLoader,所以这个写法中的MyApi.class对象实际上对应当前线程上下文classload对象,所以二者并不是同一个对象。

    正确的写法如下:

private void ResolveClass(String className) { try { Class clazz = classLoader.loadClass(className); Class aClass = classLoader.loadClass(MyApi.class.getName()); System.out.println(clazz.getName()); Annotation annotation = clazz.getAnnotation(aClass); if( annotation != null) { System.out.println(annotation.getClass().getMethod("name").invoke(annotation)); } } catch (ClassNotFoundException e) { e.printStackTrace(); } catch (NoSuchMethodException e) { e.printStackTrace(); } catch (IllegalAccessException e) { e.printStackTrace(); } catch (InvocationTargetException e) { e.printStackTrace(); } }



【本文地址】


今日新闻


推荐新闻


CopyRight 2018-2019 办公设备维修网 版权所有 豫ICP备15022753号-3