Подтвердить что ты не робот

Получить все классы в пути класса

Как я могу получить список всех доступных классов в CLASSPATH во время выполнения?
В Eclipse IDE вы можете сделать это, нажав Ctrl + Shift + T.
Есть ли какой-либо метод в Java, чтобы сделать это?

4b9b3361

Ответ 1

Вы можете получить все корни pathpath, передав пустой String в ClassLoader#getResources().

Enumeration<URL> roots = classLoader.getResources("");

Вы можете построить File на основе URL следующим образом:

File root = new File(url.getPath());

Вы можете использовать File#listFiles(), чтобы получить список всех файлов в данном каталоге:

for (File file : root.listFiles()) {
    // ...
}

Вы можете использовать стандартные java.io.File методы, чтобы проверить, является ли это каталогом и/или захватить имя файла.

if (file.isDirectory()) {
    // Loop through its listFiles() recursively.
} else {
    String name = file.getName();
    // Check if it a .class file or a .jar file and handle accordingly.
}

В зависимости от единственного функционального требования, я полагаю, что библиотека отражений намного точнее, что вы хотите.

Ответ 2

Вот что я написал, чтобы сделать это. Я уверен, что это не все, если вы делаете что-то странное с classpath, но, похоже, это работает хорошо для меня. Обратите внимание, что на самом деле он не загружает классы, он просто возвращает их имена. Это значит, что он не будет загружать все классы в память, а также потому, что некоторые классы в моей кодовой базе данных вызывают ошибки инициализации, если они загружаются в неподходящее время...

public interface Visitor<T> {
    /**
     * @return {@code true} if the algorithm should visit more results,
     * {@code false} if it should terminate now.
     */
    public boolean visit(T t);
}

public class ClassFinder {
    public static void findClasses(Visitor<String> visitor) {
        String classpath = System.getProperty("java.class.path");
        String[] paths = classpath.split(System.getProperty("path.separator"));

        String javaHome = System.getProperty("java.home");
        File file = new File(javaHome + File.separator + "lib");
        if (file.exists()) {
            findClasses(file, file, true, visitor);
        }

        for (String path : paths) {
            file = new File(path);
            if (file.exists()) {
                findClasses(file, file, false, visitor);
            }
        }
    }

    private static boolean findClasses(File root, File file, boolean includeJars, Visitor<String> visitor) {
        if (file.isDirectory()) {
            for (File child : file.listFiles()) {
                if (!findClasses(root, child, includeJars, visitor)) {
                    return false;
                }
            }
        } else {
            if (file.getName().toLowerCase().endsWith(".jar") && includeJars) {
                JarFile jar = null;
                try {
                    jar = new JarFile(file);
                } catch (Exception ex) {

                }
                if (jar != null) {
                    Enumeration<JarEntry> entries = jar.entries();
                    while (entries.hasMoreElements()) {
                        JarEntry entry = entries.nextElement();
                        String name = entry.getName();
                        int extIndex = name.lastIndexOf(".class");
                        if (extIndex > 0) {
                            if (!visitor.visit(name.substring(0, extIndex).replace("/", "."))) {
                                return false;
                            }
                        }
                    }
                }
            }
            else if (file.getName().toLowerCase().endsWith(".class")) {
                if (!visitor.visit(createClassName(root, file))) {
                    return false;
                }
            }
        }

        return true;
    }

    private static String createClassName(File root, File file) {
        StringBuffer sb = new StringBuffer();
        String fileName = file.getName();
        sb.append(fileName.substring(0, fileName.lastIndexOf(".class")));
        file = file.getParentFile();
        while (file != null && !file.equals(root)) {
            sb.insert(0, '.').insert(0, file.getName());
            file = file.getParentFile();
        }
        return sb.toString();
    }
}

Чтобы использовать его:

ClassFinder.findClasses(new Visitor<String>() {
    @Override
    public boolean visit(String clazz) {
        System.out.println(clazz)
        return true; // return false if you don't want to see any more classes
    }
});

Ответ 3

Вы можете использовать классы утилиты из пакета com.google.common.reflect из библиотеки Guava. Например. для получения всех классов в определенном пакете:

    ClassLoader cl = getClass().getClassLoader();
    Set<ClassPath.ClassInfo> classesInPackage = ClassPath.from(cl).getTopLevelClassesRecursive("com.mycompany.mypackage");

Это красноречиво, но те же предостережения, что и другие ответы, описывают, все еще применяются, а именно, что они обычно будут находить только классы, загружаемые стандартным ClassLoader, например. URLClassLoader.

Ответ 4

Каждый так часто я ищу это. Это сложно, потому что, даже если вам удастся найти все на пути к классам, вы можете не найти все доступное для данного загрузчика классов (например, я работал над проектом, который загружал определения классов непосредственно из DB однажды).

Лучше всего на этом этапе, возможно, посмотреть на Spring. Они просматривают классы в пути к классам, чтобы узнать, есть ли у них аннотации, которые им нужны для запуска kickstart.

Принятый ответ здесь - хорошее место для начала:

Сканирование аннотаций Java во время выполнения

Ответ 5

Я предлагаю использовать Spring PathMatchingResourcePatternResolver;

Он будет делать трюк как для запуска пакета из среды IDE, так и из файловой системы:

Подробнее читайте здесь:

Как получить ресурсы из пакета