Получить все классы в Classpath
Как я могу получить список всех доступных классов в CLASSPATH
во время выполнения?
В Eclipse IDE вы можете сделать это, нажав Ctrl + Shift + T.
Есть ли какой-либо метод в Java, чтобы сделать это?
5 ответов
Вы можете получить все корни classpath, передав пустой 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's a .class file or a .jar file and handle accordingly.
}
В зависимости от единственного функционального требования, я предполагаю, что библиотека Reflections - это гораздо больше, чем вы хотите.
Вот что я написал, чтобы сделать это. Я уверен, что ничего не получится, если вы делаете что-то странное с 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
}
});
Вы можете использовать служебные классы из com.google.common.reflect
пакет из библиотеки гуавы. Например, чтобы получить все классы в определенном пакете:
ClassLoader cl = getClass().getClassLoader();
Set<ClassPath.ClassInfo> classesInPackage = ClassPath.from(cl).getTopLevelClassesRecursive("com.mycompany.mypackage");
Это сжато, однако, те же предостережения, которые описывают другие ответы, все еще применимы, а именно, что он будет обычно находить только классы, загруженные "стандартным" ClassLoader, например URLClassLoader
,
Время от времени я ищу это. Это довольно сложно, потому что даже если вам удастся найти все в пути к классам, вы не сможете найти все доступное для данного загрузчика классов (например, я работал над проектом, который однажды загружал определения классов непосредственно из БД).
Лучшая ставка на данный момент - это, вероятно, посмотреть на весну. Они сканируют классы на пути к классам, чтобы увидеть, есть ли какие-нибудь аннотации, которые им нужны для запуска материала.
Принятый ответ - хорошее место для начала:
Я предлагаю использовать Spring PathMatchingResourcePatternResolver;
Это сделает трюк как для запуска пакета из IDE, так и из файловой системы:
Для более подробной информации проверьте мою заметку здесь: