java-获取类路径中的所有类

如何在运行时获取CLASSPATH中所有可用类的列表?
在Eclipse IDE中,您可以通过按Ctrl + Shift + T来执行此操作。
Java中是否有任何方法可以完成它?

arash asked 2020-08-10T15:24:46Z
5个解决方案
53 votes

您可以通过将空的java.io.File传递到java.io.File中来获取所有类路径的根。

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

您可以基于java.io.File构造java.io.File,如下所示:

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

您可以使用java.io.File来获取给定目录中所有文件的列表:

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

您可以使用标准的java.io.File方法来检查它是否是目录和/或获取文件名。

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

根据唯一的功能要求,我认为Reflections库正是您想要的。

BalusC answered 2020-08-10T15:25:09Z
20 votes

这是我写的。 我敢肯定,如果您对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
    }
});
Andy answered 2020-08-10T15:25:34Z
7 votes

您可以使用Guava库中URLClassLoader程序包中的实用程序类。 例如。 获取特定程序包中的所有类:

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

这很简洁,但是仍然适用与其他答案相同的警告,即通常只会找到由``标准''ClassLoader加载的类,例如 URLClassLoader

Matthew Wise answered 2020-08-10T15:25:59Z
0 votes

我经常找这个。 这有点困难,因为即使您设法在类路径上找到所有内容,也可能找不到对给定类加载器可用的所有内容(例如,我从事的项目是直接从数据库中一次加载类定义)。

此时最好的选择可能是研究Spring。 他们扫描类路径上的类,以查看是否有需要启动内容的注释。

此处接受的答案是一个很好的起点:

在运行时扫描Java批注

Bill K answered 2020-08-10T15:26:33Z
0 votes

我建议使用Spring的PathMatchingResourcePatternResolver;

从IDE或从文件系统启动包都将达到目的:

有关更多详细信息,请在此处查看我的注释:

如何从包中获取资源

Naor Bar answered 2020-08-10T15:27:08Z
translate from https://stackoverflow.com:/questions/3222638/get-all-of-the-classes-in-the-classpath