1. 程式人生 > >全棧的自我修養: 0005 Java 包掃描實現和應用(Jar篇)

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

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

> It's not the altitude, it's the attitude.
> 決定一切的不是高度而是態度。
![](https://img2020.cnblogs.com/blog/1246875/202007/1246875-20200724221028563-62249441.png) **Table of Contents** - [依賴的 Jar](#依賴的-jar) - [思路](#思路) - [完整程式碼](#完整程式碼) - [整合後代碼](#整合後代碼) 如果你曾經使用過 `Spring`, 那你已經配過 包掃描路徑吧,那包掃描是怎麼實現的呢?讓我們自己寫個包掃描 上篇文章中介紹了使用 `File` 遍歷的方式去進行包掃描,這篇主要補充一下`jar`包的掃描方式,在我們的專案中一般都會去依賴一些其他`jar` 包, 比如新增 guava 依賴 ```xml com.google.guava
guava 28.2-jre
``` 我們再次執行上次的測試用例 ```java @Test public void testGetPackageAllClasses() throws IOException, ClassNotFoundException { ClassScanner scanner = new ClassScanner("com.google.common.cache", true, null, null); Set> packageAllClasses = scanner.doScanAllClasses(); packageAllClasses.forEach(it -> { System.out.println(it.getName()); }); } ``` 什麼都沒有輸出 # 依賴的 Jar 基於`Java` 的反射機制,我們很容易根據 `class` 去建立一個例項物件,但如果我們根本不知道某個包下有多少物件時,我們應該怎麼做呢? 在使用`Spring`框架時,會根據包掃描路徑來找到所有的 `class`, 並將其例項化後存入容器中。 在我們的專案中也會遇到這樣的場景,比如某個包為 `org.example.plugins`, 這個裡面放著所有的外掛,為了不每次增減外掛都要手動修改程式碼,我們可能會想到用掃描的方式去動態獲知 `org.example.plugins` 到底有多少 class, 當然應用場景很有很多 # 思路 既然知道是採用了 `jar` , 那我們使用遍歷 jar 的方式去處理一下 ```java JarFile jar = ((JarURLConnection) url.openConnection()).getJarFile(); // 遍歷jar包中的元素 Enumeration entries = jar.entries(); while (entries.hasMoreElements()) { JarEntry entry = entries.nextElement(); String name = entry.getName(); } ``` 這裡獲取的name 格式為 `com/google/common/cache/Cache.class` 是不是和上篇的檔案路徑很像呀, 這裡可以通過對 `name` 進行操作獲取`包名`和 `class` ```java // 獲取包名 String jarPackageName = name.substring(0, name.lastIndexOf('/')).replace("/", "."); // 獲取 class 路徑, 這樣就能通過類載入進行載入了 String className = name.replace('/', '.'); className = className.substring(0, className.length() - 6); ``` # 完整程式碼 ```java private void doScanPackageClassesByJar(String basePackage, URL url, Set> classes) throws IOException, ClassNotFoundException { // 包名 String packageName = basePackage; // 獲取檔案路徑 String basePackageFilePath = packageName.replace('.', '/'); // 轉為jar包 JarFile jar = ((JarURLConnection) url.openConnection()).getJarFile(); // 遍歷jar包中的元素 Enumeration 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` 掃描方式就是完成的程式碼了 # 整合後代碼 ```java 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 packagePredicate; private final Predicate 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 packagePredicate, Predicate 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> doScanAllClasses() throws IOException, ClassNotFoundException { Set> classes = new LinkedHashSet>(); String packageName = basePackage; // 如果最後一個字元是“.”,則去掉 if (packageName.endsWith(".")) { packageName = packageName.substring(0, packageName.lastIndexOf('.')); } // 將包名中的“.”換成系統資料夾的“/” String basePackageFilePath = packageName.replace('.', '/'); Enumeration 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> classes) throws IOException, ClassNotFoundException { // 包名 String packageName = basePackage; // 獲取檔案路徑 String basePackageFilePath = packageName.replace('.', '/'); // 轉為jar包 JarFile jar = ((JarURLConnection) url.openConnection()).getJarFile(); // 遍歷jar包中的元素 Enumeration 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> 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); } } } }