All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.xiongyingqi.common.utils.scan.ClassLookupHelper Maven / Gradle / Ivy

The newest version!
package com.xiongyingqi.common.utils.scan;

import java.io.File;
import java.io.FileFilter;
import java.io.IOException;
import java.net.JarURLConnection;
import java.net.URL;
import java.net.URLClassLoader;
import java.util.*;
import java.util.jar.JarEntry;
import java.util.jar.JarFile;

/**
 * @author xiongyingqi
 * @since 20171019//
 */
public class ClassLookupHelper {
  /**
   * Class File 过滤器
   */
  public static interface ClassFileFilter {

    /**
     * 过滤磁盘文件
     */
    public boolean accept(String klassName, File file, ClassLoader loader);

    /**
     * 过滤 Jar 包中的文件
     */
    public boolean accept(String klassName, JarFile jar, JarEntry entry, ClassLoader loader);
  }

  /**
   * 从所有的 classpath 下面搜索指定的 Class
   */
  public static Collection> getClasses(ClassFileFilter filter) {
    Set loaders = new LinkedHashSet(8);
    loaders.addAll(getClassLoaders(Thread.currentThread().getContextClassLoader()));
    loaders.addAll(getClassLoaders(ClassLookupHelper.class.getClassLoader()));

    Set> klasses = new LinkedHashSet>();
    for (URLClassLoader cl : loaders) {
      for (URL url : cl.getURLs()) {
        String file = url.getFile();
        if (file.endsWith(".jar") || file.endsWith(".zip")) {
          lookupClassesInJar(null, url, true, cl, filter, klasses);
        } else {
          lookupClassesInFileSystem(null, new File(file), true, cl, filter, klasses);
        }
      }
    }
    return klasses;
  }

  // 根据 baseClassLoader 找到所有的祖先 URLClassLoader (包括自己)
  private static Collection getClassLoaders(ClassLoader baseClassLoader) {
    Collection loaders = new ArrayList(8);
    ClassLoader loader = baseClassLoader;
    while (loader != null) {
      if ("sun.misc.Launcher$ExtClassLoader".equals(loader.getClass().getName())) {
        break;
      }
      if (loader instanceof URLClassLoader) {
        loaders.add((URLClassLoader) loader);
      }
      loader = loader.getParent();
    }
    return loaders;
  }


  /**
   * 从指定 package 中获取所有的 Class
   *
   * @param pkg       包
   * @param recursive 是否递归查找
   * @param filter    class 过滤器
   * @return 所有找到的 Class
   */
  public static Set> getClasses(Package pkg, boolean recursive, ClassFileFilter filter) {
    return getClasses(pkg.getName(), recursive, filter);
  }

  /**
   * 从指定 package 中获取所有的 Class
   *
   * @param packageName 包名
   * @param recursive   是否递归查找
   * @param filter      class 过滤器
   * @return 所有找到的 Class
   */
  public static Set> getClasses(String packageName, boolean recursive, ClassFileFilter filter) {
    if (packageName == null || packageName.length() == 0) {
      throw new IllegalArgumentException("packageName is empty.");
    }

    ClassLoader loader = Thread.currentThread().getContextClassLoader();
    String packageDirName = packageName.replace('.', '/');
    Collection urls;
    try {
      Enumeration dirs = loader.getResources(packageDirName);
      urls = Collections.list(dirs);
    } catch (IOException e) {
      throw new RuntimeException(e);
    }

    Set> klasses = new LinkedHashSet>();
    for (URL url : urls) {
      String protocol = url.getProtocol();
      if ("file".equals(protocol)) {
        lookupClassesInFileSystem(packageName, new File(url.getFile()), recursive, loader, filter, klasses);
      } else if ("jar".equals(protocol)) {
        lookupClassesInJar(packageName, url, recursive, loader, filter, klasses);
      }
    }
    return klasses;
  }

  /**
   * 以文件的形式来获取包下的所有 Class
   *
   * @param packageName JAVA包名
   * @param packagePath 包所在的文件目录
   * @param recursive   是否递归查找
   * @param loader      负责该包的 ClassLoader
   * @param filter      class 过滤器
   * @param klasses     返回找到的 class
   */
  private static void lookupClassesInFileSystem(String packageName, File packagePath, final boolean recursive, ClassLoader loader, ClassFileFilter filter, Set> klasses) {
    if (!packagePath.exists() || !packagePath.isDirectory()) {
      return;
    }
    File[] dirfiles = packagePath.listFiles(new FileFilter() {
      public boolean accept(File file) {
        return (recursive && file.isDirectory()) || (file.getName().endsWith(".class"));
      }
    });

    String packageNamePrefix = "";
    if (packageName != null && packageName.length() > 0) {
      packageNamePrefix = packageName + '.';
    }
    for (File file : dirfiles) {
      if (file.isDirectory()) {
        lookupClassesInFileSystem(packageNamePrefix + file.getName(), file, recursive, loader, filter, klasses);
      } else {
        // 去掉后面的 .class 只留下类名
        String klassName = packageNamePrefix + file.getName().substring(0, file.getName().length() - 6);
        try {
          if (filter == null || filter.accept(klassName, file, loader)) {
            Class klass = loader.loadClass(klassName);
            klasses.add(klass);
          }
        } catch (Throwable e) {
        }
      }
    }
  }

  /**
   * 以在 Jar 包中获取指定包下的所有 Class
   *
   * @param packageName JAVA包名
   * @param jarUrl      Jar    包文件对应的 URL
   * @param recursive   是否递归查找
   * @param loader      负责该包的 ClassLoader
   * @param filter      class 过滤器
   * @param klasses     返回找到的 class
   */
  private static void lookupClassesInJar(String packageName, URL jarUrl, boolean recursive, ClassLoader loader, ClassFileFilter filter, Set> klasses) {
    String packageDirName = "";
    if (packageName != null && packageName.length() > 0) {
      packageDirName = packageName.replace('.', '/') + '/';
    }

    JarFile jar = null;
    try {
      if ("jar".equals(jarUrl.getProtocol())) {
        jar = ((JarURLConnection) jarUrl.openConnection()).getJarFile();
      } else {
        jar = new JarFile(jarUrl.getFile());
      }
      Enumeration entries = jar.entries();
      while (entries.hasMoreElements()) {
        // 获取jar里的一个实体 可以是目录和一些jar包里的其他文件 如META-INF等文件
        JarEntry entry = entries.nextElement();
        if (entry.isDirectory()) {
          continue;
        }

        String name = entry.getName();
        if (name.charAt(0) == '/') {
          name = name.substring(1);
        }
        if (name.startsWith(packageDirName) && name.endsWith(".class")) {
          if (name.lastIndexOf('/') > packageDirName.length()) {
            // 在子包内
            if (!recursive) {
              continue;
            }
          }
          // 去掉后面的 .class 只留下类名
          String klassName = name.substring(0, name.length() - 6);
          klassName = klassName.replace('/', '.');
          try {
            if (filter == null || filter.accept(klassName, jar, entry, loader)) {
              Class klass = loader.loadClass(klassName);
              klasses.add(klass);
            }
          } catch (Throwable e) {
          }
        }
      }
    } catch (IOException e) {
    } finally {
    }
  }

  public static void main(String[] args) {
    Collection> classes = getClasses(new ClassFileFilter() {
      @Override
      public boolean accept(String klassName, File file, ClassLoader loader) {
        return true;
      }

      @Override
      public boolean accept(String klassName, JarFile jar, JarEntry entry, ClassLoader loader) {
        return true;
      }
    });
    for (Class aClass : classes) {
      System.out.println(aClass);
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy