无法从Java类获取自定义注释

问题描述 投票:1回答:1

我想从Java类获取类级别的注释:

    class FixAnnotation {

        public String[] author;

    }

    @Retention(RetentionPolicy.RUNTIME)
    @Target(ElementType.TYPE)
    public @interface Fix {

        public String[] author() default "";

    }

我尝试了这个已编译的Java类的示例:

@Component("test")
@Fix(
    author = {"Example author 1", "Example author 2"}
)
public class Order implements Action {
..
}

但是当我尝试时:

公共列表listLocalFilesAndDirsAllLevels(File baseDir){

        List<File>  collectedFilesAndDirs = new ArrayList<>();
        Deque<File> remainingDirs = new ArrayDeque<>();

        if(baseDir.exists()) {
            remainingDirs.add(baseDir);

            while(!remainingDirs.isEmpty()) {
                File dir = remainingDirs.removeLast();
                List<File> filesInDir = Arrays.asList(dir.listFiles());
                for(File fileOrDir : filesInDir)  {
                    // We need to process only .class files
                    if(fileOrDir.getName().endsWith(".class")){
                        collectedFilesAndDirs.add(fileOrDir);
                        if(fileOrDir.isDirectory()) {
                            remainingDirs.add(fileOrDir);
                        }
                    }
                }
            }
        }

        return collectedFilesAndDirs;
    }

      List<File> list;

      for(int i=0; i<list.size(); i++) {
            File item = list.get(i);
            System.out.println(item.getName());

            Fix name = item.getClass().getAnnotation(Fix.class);

            out.println("author: " + name.author());
       }

我得到了NPE。您知道我如何获取注释内容吗?

编辑:我试过了:

public static void main() throws Exception
        {    
            final File folder = new File("/opt/test");
            processAnnotatedFiles(listLocalFilesAndDirsAllLevels(folder));

        }

    public void processAnnotatedFiles(List<File> list) throws IOException, ClassNotFoundException {
        out.println("Directory files size " + list.size());

        for(int i=0; i<list.size(); i++) {
            out.println("File " + list.get(i).getName());

            File file = list.get(i);

            String path = file.getPath();

            String[] authors = getFixFromClassFile(Paths.get(path));
            System.out.println(Arrays.toString(authors));
        }

    }

    public List<File> listLocalFilesAndDirsAllLevels(File baseDir) {

        List<File>  collectedFilesAndDirs = new ArrayList<>();
        Deque<File> remainingDirs = new ArrayDeque<>();

        if(baseDir.exists()) {
            remainingDirs.add(baseDir);

            while(!remainingDirs.isEmpty()) {
                File dir = remainingDirs.removeLast();
                List<File> filesInDir = Arrays.asList(dir.listFiles());
                for(File fileOrDir : filesInDir)  {
                    // We need to process only .class files
                    if(fileOrDir.getName().endsWith(".class")){
                        collectedFilesAndDirs.add(fileOrDir);
                        if(fileOrDir.isDirectory()) {
                            remainingDirs.add(fileOrDir);
                        }
                    }
                }
            }
        }

        return collectedFilesAndDirs;
    }

    private String[] getFixFromClassFile(Path pathToClass) throws MalformedURLException, ClassNotFoundException {
        // Create class loader based on path
        URLClassLoader loader = new URLClassLoader(new URL[]{pathToClass.toUri().toURL()});

        // convert path to class with package
        String classWithPackage = getClassWithPackageFromPath(pathToClass);

        // Load class dynamically
        Class<?> clazz = loader.loadClass(classWithPackage);
        Fix fix = clazz.getAnnotation(Fix.class);
        if (fix == null) {
            return new String[0];
        }

        return fix.author();
    }

    private String getClassWithPackageFromPath(Path pathToClass) {
        final String packageStartsFrom = "com.";
        final String classFileExtension = ".class";
        final String pathWithDots = pathToClass.toString().replace(File.separator, ".");
        return pathWithDots.substring(pathWithDots.indexOf(packageStartsFrom)).replace(classFileExtension, "");
    }

我得到java.lang.StringIndexOutOfBoundsException: String index out of range: -1 at java.lang.String.substring(String.java:1927)

java reflection annotations
1个回答
1
投票

当您在getClass对象上调用File方法时,它将返回java.io.File Class实例。此方法不会从给定文件加载类。

如果要从给定的*.class文件中加载类,则需要使用java.lang.ClassLoader实现。例如,java.net.URLClassLoader。在下面,您可以找到如何加载类并检查注释的示例:

import java.io.File;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
import java.net.MalformedURLException;
import java.net.URL;
import java.net.URLClassLoader;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.Arrays;

@Fix(author = "Test author")
public class ReflectionApp {

    public static void main(String[] args) throws Exception {
        String path = "path/to/com/so/ReflectionApp.class";
        String[] authors = getFixFromClassFile(Paths.get(path));
        System.out.println(Arrays.toString(authors));
    }

    private static String[] getFixFromClassFile(Path pathToClass) throws MalformedURLException, ClassNotFoundException {
        // Create class loader based on path
        URLClassLoader loader = new URLClassLoader(new URL[]{pathToClass.toUri().toURL()});

        // convert path to class with package
        String classWithPackage = getClassWithPackageFromPath(pathToClass);

        // Load class dynamically
        Class<?> clazz = loader.loadClass(classWithPackage);
        Fix fix = clazz.getAnnotation(Fix.class);
        if (fix == null) {
            return new String[0];
        }

        return fix.author();
    }

    private static String getClassWithPackageFromPath(Path pathToClass) {
        final String packageStartsFrom = "com.";
        final String classFileExtension = ".class";
        final String pathWithDots = pathToClass.toString().replace(File.separator, ".");
        return pathWithDots.substring(pathWithDots.indexOf(packageStartsFrom)).replace(classFileExtension, "");
    }
}

以上代码打印:

[Test author]

另请参见:

© www.soinside.com 2019 - 2024. All rights reserved.