5

パッケージ内のすべてのクラスを取得するにはどうすればよいですか?

4

6 に答える 6

10

できません。クラスは、リモートのものを含め、さまざまなクラスローダーを介して取得できます。

于 2009-07-20T23:40:28.753 に答える
7

JGによって投稿されたアイデアに基づいて、これをjarに対して解決するためのより完全な方法を次に示します。

/**
 * Scans all classloaders for the current thread for loaded jars, and then scans
 * each jar for the package name in question, listing all classes directly under
 * the package name in question. Assumes directory structure in jar file and class
 * package naming follow java conventions (i.e. com.example.test.MyTest would be in
 * /com/example/test/MyTest.class)
 */
public Collection<Class> getClassesForPackage(String packageName) throws Exception {
  String packagePath = packageName.replace(".", "/");
  ClassLoader classLoader = Thread.currentThread().getContextClassLoader();
  Set<URL> jarUrls = new HashSet<URL>();

  while (classLoader != null) {
    if (classLoader instanceof URLClassLoader)
      for (URL url : ((URLClassLoader) classLoader).getURLs())
        if (url.getFile().endsWith(".jar")  // may want better way to detect jar files
          jarUrls.add(url);

    classLoader = classLoader.getParent();
  }

  Set<Class> classes = new HashSet<Class>();

  for (URL url : jarUrls) {
    JarInputStream stream = new JarInputStream(url.openStream()); // may want better way to open url connections
    JarEntry entry = stream.getNextJarEntry();

    while (entry != null) {
      String name = entry.getName();
      int i = name.lastIndexOf("/");

      if (i > 0 && name.endsWith(".class") && name.substring(0, i).equals(packagePath)) 
        classes.add(Class.forName(name.substring(0, name.length() - 6).replace("/", ".")));

      entry = stream.getNextJarEntry();
    }

    stream.close();
  }

  return classes;
}
于 2009-07-21T04:46:07.987 に答える
4

クラスがローカルで見つかると仮定して、まさにあなたが望むことをするここからのスニペットがあります:

private static Class[] getClasses(String packageName)
throws ClassNotFoundException, IOException {
    ClassLoader classLoader = Thread.currentThread().getContextClassLoader();
    assert classLoader != null;
    String path = packageName.replace('.', '/');
    Enumeration<URL> resources = classLoader.getResources(path);
    List<File> dirs = new ArrayList<File>();
    while (resources.hasMoreElements()) {
        URL resource = resources.nextElement();
        dirs.add(new File(resource.getFile()));
    }
    ArrayList<Class> classes = new ArrayList<Class>();
    for (File directory : dirs) {
        classes.addAll(findClasses(directory, packageName));
    }
    return classes.toArray(new Class[classes.size()]);
}

private static List<Class> findClasses(File directory, String packageName) throws ClassNotFoundException {
    List<Class> classes = new ArrayList<Class>();
    if (!directory.exists()) {
        return classes;
    }
    File[] files = directory.listFiles();
    for (File file : files) {
        if (file.isDirectory()) {
            assert !file.getName().contains(".");
            classes.addAll(findClasses(file, packageName + "." + file.getName()));
        } else if (file.getName().endsWith(".class")) {
            classes.add(Class.forName(packageName + '.' + file.getName().substring(0, file.getName().length() - 6)));
        }
    }
    return classes;
}
于 2009-07-20T23:46:33.780 に答える
1

上記の回答は、必要な機能の概要を示しています。ただし、このテーマの詳細については、こちらこちらをご覧ください。

于 2009-07-21T18:34:50.003 に答える
1

それを行うグローバルな方法はありません。そうは言っても、クラスがどこから来ているかがわかっている場合は、jar ファイルまたはファイル システムのディレクトリをたどることができます。

于 2009-07-20T23:44:56.687 に答える
1

Java には検出機能がありません。

新しいクラスを追加 (検出) できるほとんどの製品には、「プログラム拡張機能」を説明するテキスト ファイル、または @JG が説明したようなトリックを使用するクラスまたは jar を配置できる特定のディレクトリがあります。(これはEclipseが行うことであり、ユーザーが手動で新しいモジュールを追加できるソリューションに推奨されます)

于 2009-07-20T23:56:23.013 に答える