全棧的自我修養: 0005 Java 包掃描實現和應用(Jar篇)

全棧的自我修養: 0005 Java 包掃描實現和應用(Jar篇)

It's not the altitude, it's the attitude.

決定一切的不是高度而是態度。
java

Table of Contents框架

若是你曾經使用過 Spring, 那你已經配過 包掃描路徑吧,那包掃描是怎麼實現的呢?讓咱們本身寫個包掃描測試

上篇文章中介紹了使用 File 遍歷的方式去進行包掃描,這篇主要補充一下jar包的掃描方式,在咱們的項目中通常都會去依賴一些其餘jar 包,this

好比添加 guava 依賴google

<dependency>
    <groupId>com.google.guava</groupId>
    <artifactId>guava</artifactId>
    <version>28.2-jre</version>
</dependency>

咱們再次運行上次的測試用例url

@Test
public void testGetPackageAllClasses() throws IOException, ClassNotFoundException {
    ClassScanner scanner = new ClassScanner("com.google.common.cache", true, null, null);
    Set<Class<?>> packageAllClasses = scanner.doScanAllClasses();
    packageAllClasses.forEach(it -> {
        System.out.println(it.getName());
    });
}

什麼都沒有輸出.net

依賴的 Jar

基於Java 的反射機制,咱們很容易根據 class 去建立一個實例對象,但若是咱們根本不知道某個包下有多少對象時,咱們應該怎麼作呢?插件

在使用Spring框架時,會根據包掃描路徑來找到全部的 class, 並將其實例化後存入容器中。線程

在咱們的項目中也會遇到這樣的場景,好比某個包爲 org.example.plugins, 這個裏面放着全部的插件,爲了避免每次增減插件都要手動修改代碼,咱們可能會想到用掃描的方式去動態獲知 org.example.plugins 到底有多少 class, 固然應用場景頗有不少code

思路

既然知道是採用了 jar , 那咱們使用遍歷 jar 的方式去處理一下

JarFile jar = ((JarURLConnection) url.openConnection()).getJarFile();
// 遍歷jar包中的元素
Enumeration<JarEntry> entries = jar.entries();

while (entries.hasMoreElements()) {
  JarEntry entry = entries.nextElement();
  String name = entry.getName();
}

這裏獲取的name 格式爲 com/google/common/cache/Cache.class 是否是和上篇的文件路徑很像呀, 這裏能夠經過對 name 進行操做獲取包名class

// 獲取包名
String jarPackageName = name.substring(0, name.lastIndexOf('/')).replace("/", ".");

// 獲取 class 路徑, 這樣就能經過類加載進行加載了
String className = name.replace('/', '.');
className = className.substring(0, className.length() - 6);

完整代碼

private void doScanPackageClassesByJar(String basePackage, URL url, Set<Class<?>> classes)
    throws IOException, ClassNotFoundException {
  // 包名
  String packageName = basePackage;
  // 獲取文件路徑
  String basePackageFilePath = packageName.replace('.', '/');
  // 轉爲jar包
  JarFile jar = ((JarURLConnection) url.openConnection()).getJarFile();
  // 遍歷jar包中的元素
  Enumeration<JarEntry> entries = jar.entries();
  while (entries.hasMoreElements()) {
    JarEntry entry = entries.nextElement();
    String name = entry.getName();
    // 若是路徑不一致,或者是目錄,則繼續
    if (!name.startsWith(basePackageFilePath) || entry.isDirectory()) {
      continue;
    }
    // 判斷是否遞歸搜索子包
    if (!recursive && name.lastIndexOf('/') != basePackageFilePath.length()) {
      continue;
    }

    if (packagePredicate != null) {
      String jarPackageName = name.substring(0, name.lastIndexOf('/')).replace("/", ".");
      if (!packagePredicate.test(jarPackageName)) {
        continue;
      }
    }

    // 斷定是否符合過濾條件
    String className = name.replace('/', '.');
    className = className.substring(0, className.length() - 6);
    // 用當前線程的類加載器加載類
    Class<?> loadClass = Thread.currentThread().getContextClassLoader().loadClass(className);
    if (classPredicate == null || classPredicate.test(loadClass)) {
      classes.add(loadClass);
    }

  }
}

在結合上篇中 File 掃描方式就是完成的代碼了

整合後代碼

package org.example;

import java.io.File;
import java.io.FileFilter;
import java.io.IOException;
import java.net.JarURLConnection;
import java.net.URL;
import java.net.URLDecoder;
import java.util.Enumeration;
import java.util.LinkedHashSet;
import java.util.Set;
import java.util.function.Predicate;
import java.util.jar.JarEntry;
import java.util.jar.JarFile;

/**
 * class 掃描器
 *
 * @author zhangyunan
 */
public class ClassScanner {

  private final String basePackage;
  private final boolean recursive;
  private final Predicate<String> packagePredicate;
  private final Predicate<Class> classPredicate;


  /**
   * Instantiates a new Class scanner.
   *
   * @param basePackage      the base package
   * @param recursive        是否遞歸掃描
   * @param packagePredicate the package predicate
   * @param classPredicate   the class predicate
   */
  public ClassScanner(String basePackage, boolean recursive, Predicate<String> packagePredicate,
    Predicate<Class> classPredicate) {
    this.basePackage = basePackage;
    this.recursive = recursive;
    this.packagePredicate = packagePredicate;
    this.classPredicate = classPredicate;
  }

  /**
   * Do scan all classes set.
   *
   * @return the set
   * @throws IOException            the io exception
   * @throws ClassNotFoundException the class not found exception
   */
  public Set<Class<?>> doScanAllClasses() throws IOException, ClassNotFoundException {

    Set<Class<?>> classes = new LinkedHashSet<Class<?>>();

    String packageName = basePackage;

    // 若是最後一個字符是「.」,則去掉
    if (packageName.endsWith(".")) {
      packageName = packageName.substring(0, packageName.lastIndexOf('.'));
    }

    // 將包名中的「.」換成系統文件夾的「/」
    String basePackageFilePath = packageName.replace('.', '/');

    Enumeration<URL> resources = Thread.currentThread().getContextClassLoader().getResources(basePackageFilePath);
    while (resources.hasMoreElements()) {
      URL resource = resources.nextElement();
      String protocol = resource.getProtocol();
      if ("file".equals(protocol)) {
        String filePath = URLDecoder.decode(resource.getFile(), "UTF-8");
        // 掃描文件夾中的包和類
        doScanPackageClassesByFile(classes, packageName, filePath);
      } else if ("jar".equals(protocol)) {
        doScanPackageClassesByJar(packageName, resource, classes);
      }
    }

    return classes;
  }

  private void doScanPackageClassesByJar(String basePackage, URL url, Set<Class<?>> classes)
    throws IOException, ClassNotFoundException {
    // 包名
    String packageName = basePackage;
    // 獲取文件路徑
    String basePackageFilePath = packageName.replace('.', '/');
    // 轉爲jar包
    JarFile jar = ((JarURLConnection) url.openConnection()).getJarFile();
    // 遍歷jar包中的元素
    Enumeration<JarEntry> entries = jar.entries();
    while (entries.hasMoreElements()) {
      JarEntry entry = entries.nextElement();
      String name = entry.getName();
      // 若是路徑不一致,或者是目錄,則繼續
      if (!name.startsWith(basePackageFilePath) || entry.isDirectory()) {
        continue;
      }
      // 判斷是否遞歸搜索子包
      if (!recursive && name.lastIndexOf('/') != basePackageFilePath.length()) {
        continue;
      }

      if (packagePredicate != null) {
        String jarPackageName = name.substring(0, name.lastIndexOf('/')).replace("/", ".");
        if (!packagePredicate.test(jarPackageName)) {
          continue;
        }
      }

      // 斷定是否符合過濾條件
      String className = name.replace('/', '.');
      className = className.substring(0, className.length() - 6);
      // 用當前線程的類加載器加載類
      Class<?> loadClass = Thread.currentThread().getContextClassLoader().loadClass(className);
      if (classPredicate == null || classPredicate.test(loadClass)) {
        classes.add(loadClass);
      }

    }
  }

  /**
   * 在文件夾中掃描包和類
   */
  private void doScanPackageClassesByFile(Set<Class<?>> classes, String packageName, String packagePath)
    throws ClassNotFoundException {
    // 轉爲文件
    File dir = new File(packagePath);
    if (!dir.exists() || !dir.isDirectory()) {
      return;
    }
    // 列出文件,進行過濾
    // 自定義文件過濾規則
    File[] dirFiles = dir.listFiles((FileFilter) file -> {
      String filename = file.getName();

      if (file.isDirectory()) {
        if (!recursive) {
          return false;
        }

        if (packagePredicate != null) {
          return packagePredicate.test(packageName + "." + filename);
        }
        return true;
      }

      return filename.endsWith(".class");
    });

    if (null == dirFiles) {
      return;
    }

    for (File file : dirFiles) {
      if (file.isDirectory()) {
        // 若是是目錄,則遞歸
        doScanPackageClassesByFile(classes, packageName + "." + file.getName(), file.getAbsolutePath());
      } else {
        // 用當前類加載器加載 去除 fileName 的 .class 6 位
        String className = file.getName().substring(0, file.getName().length() - 6);
        Class<?> loadClass = Thread.currentThread().getContextClassLoader().loadClass(packageName + '.' + className);
        if (classPredicate == null || classPredicate.test(loadClass)) {
          classes.add(loadClass);
        }
      }
    }
  }
}
相關文章
相關標籤/搜索