今天写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();
}
}