全棧的自我修養: 0005 Java 包掃描實現和應用(Jar篇)
It's not the altitude, it's the attitude.
決定一切的不是高度而是態度。
Table of Contents
如果你曾經使用過 Spring
, 那你已經配過 包掃描路徑吧,那包掃描是怎麼實現的呢?讓我們自己寫個包掃描
上篇文章中介紹了使用 File
遍歷的方式去進行包掃描,這篇主要補充一下jar
包的掃描方式,在我們的專案中一般都會去依賴一些其他jar
包,
比如新增 guava 依賴
<dependency>
<groupId>com.google.guava</groupId>
<artifactId>guava</artifactId>
<version>28.2-jre</version>
</dependency>
我們再次執行上次的測試用例
@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());
});
}
什麼都沒有輸出
依賴的 Jar
基於Java
的反射機制,我們很容易根據 class
去建立一個例項物件,但如果我們根本不知道某個包下有多少物件時,我們應該怎麼做呢?
在使用Spring
框架時,會根據包掃描路徑來找到所有的 class
, 並將其例項化後存入容器中。
在我們的專案中也會遇到這樣的場景,比如某個包為 org.example.plugins
, 這個裡面放著所有的外掛,為了不每次增減外掛都要手動修改程式碼,我們可能會想到用掃描的方式去動態獲知 org.example.plugins
到底有多少 class, 當然應用場景很有很多
思路
既然知道是採用了 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);
}
}
}
}
}