扫描指定包下的类(包括jar文件里的类)

[color=red]blog迁移至[/color]:[url=http://www.micmiu.com]http://www.micmiu.com[/url]

本文永久链接:[url=http://sjsky.iteye.com/blog/1061092]http://sjsky.iteye.com/blog/1061092[/url]

一个实用的处理类,主要实现的功能是:自动扫描指定包下的所有类,[color=red]可以设置是否扫描内部类、是否递归扫描子包、自定义过滤规则等等[/color],下面介绍基本用法及其代码:

[color=blue][size=medium]一、方法说明[/size][/color]

// 无参构造器,默认是排除内部类、并搜索符合规则
ClassPathScanHandler handler =new ClassPathScanHandler();

// 创建一个扫描处理器,排除内部类 扫描符合条件的类
ClassPathScanHandler handler = new ClassPathScanHandler(true, true,
classFilters);

//递归扫描包:org.apache.commons.io 下符合自定义过滤规则的类
Set<Class<?>> calssList = handler.getPackageAllClasses(
"org.apache.commons.io", true);


[color=blue][size=medium]二、参数说明[/size][/color]
[list]
[*]boolean excludeInner:[color=red]该参数表示是否排除内部类,true->是 false->否,默认=true[/color]
[*]boolean checkInOrEx:[color=red]该参数表示过滤规则适用情况,true—>搜索符合规则的 false->排除符合规则的,默认=true[/color]
[*]List<String> classFilters:[color=red]自定义过滤规则,如果是null或者空,即全部符合不过滤,默认为null。过滤规则可以自定义为:Xyz或Xyz*或*Xyz或*Xyz*等类似的格式[/color]
[*]Set<Class<?>> getPackageAllClasses(String basePackage,boolean recursive);[color=red]参数 basePackage:标识扫描的基包 ; 参数:recursive:标识是否递归扫描子包[/color]
[/list]

[color=blue][size=medium]三、处理类代码[/size][/color]
ClassPathScanHandler.java

package michael.utils;

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.ArrayList;
import java.util.Enumeration;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Set;
import java.util.jar.JarEntry;
import java.util.jar.JarFile;
import java.util.regex.Pattern;

import org.apache.log4j.Logger;

/**
* 扫描指定包(包括jar)下的class文件 <br>
* <a href="http://sjsky.iteye.com">http://sjsky.iteye.com</a>
* @author michael
*/
public class ClassPathScanHandler {

/**
* logger
*/
private static final Logger logger = Logger
.getLogger(ClassPathScanHandler.class);

/**
* 是否排除内部类 true->是 false->否
*/
private boolean excludeInner = true;
/**
* 过滤规则适用情况 true—>搜索符合规则的 false->排除符合规则的
*/
private boolean checkInOrEx = true;

/**
* 过滤规则列表 如果是null或者空,即全部符合不过滤
*/
private List<String> classFilters = null;

/**
* 无参构造器,默认是排除内部类、并搜索符合规则
*/
public ClassPathScanHandler() {
}

/**
* excludeInner:是否排除内部类 true->是 false->否<br>
* checkInOrEx:过滤规则适用情况 true—>搜索符合规则的 false->排除符合规则的<br>
* classFilters:自定义过滤规则,如果是null或者空,即全部符合不过滤
* @param excludeInner
* @param checkInOrEx
* @param classFilters
*/
public ClassPathScanHandler(Boolean excludeInner, Boolean checkInOrEx,
List<String> classFilters) {
this.excludeInner = excludeInner;
this.checkInOrEx = checkInOrEx;
this.classFilters = classFilters;

}

/**
* 扫描包
* @param basePackage 基础包
* @param recursive 是否递归搜索子包
* @return Set
*/
public Set<Class<?>> getPackageAllClasses(String basePackage,
boolean recursive) {
Set<Class<?>> classes = new LinkedHashSet<Class<?>>();
String packageName = basePackage;
if (packageName.endsWith(".")) {
packageName = packageName
.substring(0, packageName.lastIndexOf('.'));
}
String package2Path = packageName.replace('.', '/');

Enumeration<URL> dirs;
try {
dirs = Thread.currentThread().getContextClassLoader().getResources(
package2Path);
while (dirs.hasMoreElements()) {
URL url = dirs.nextElement();
String protocol = url.getProtocol();
if ("file".equals(protocol)) {
logger.info("扫描file类型的class文件....");
String filePath = URLDecoder.decode(url.getFile(), "UTF-8");
doScanPackageClassesByFile(classes, packageName, filePath,
recursive);
} else if ("jar".equals(protocol)) {
logger.info("扫描jar文件中的类....");
doScanPackageClassesByJar(packageName, url, recursive,
classes);
}
}
} catch (IOException e) {
logger.error("IOException error:", e);
}

return classes;
}

/**
* 以jar的方式扫描包下的所有Class文件<br>
* @param basePackage eg:michael.utils.
* @param url
* @param recursive
* @param classes
*/
private void doScanPackageClassesByJar(String basePackage, URL url,
final boolean recursive, Set<Class<?>> classes) {
String packageName = basePackage;
String package2Path = packageName.replace('.', '/');
JarFile jar;
try {
jar = ((JarURLConnection) url.openConnection()).getJarFile();
Enumeration<JarEntry> entries = jar.entries();
while (entries.hasMoreElements()) {
JarEntry entry = entries.nextElement();
String name = entry.getName();
if (!name.startsWith(package2Path) || entry.isDirectory()) {
continue;
}

// 判断是否递归搜索子包
if (!recursive
&& name.lastIndexOf('/') != package2Path.length()) {
continue;
}
// 判断是否过滤 inner class
if (this.excludeInner && name.indexOf('$') != -1) {
logger.info("exclude inner class with name:" + name);
continue;
}
String classSimpleName = name
.substring(name.lastIndexOf('/') + 1);
// 判定是否符合过滤条件
if (this.filterClassName(classSimpleName)) {
String className = name.replace('/', '.');
className = className.substring(0, className.length() - 6);
try {
classes.add(Thread.currentThread()
.getContextClassLoader().loadClass(className));
} catch (ClassNotFoundException e) {
logger.error("Class.forName error:", e);
}
}
}
} catch (IOException e) {
logger.error("IOException error:", e);
}
}

/**
* 以文件的方式扫描包下的所有Class文件
*
* @param packageName
* @param packagePath
* @param recursive
* @param classes
*/
private void doScanPackageClassesByFile(Set<Class<?>> classes,
String packageName, String packagePath, boolean recursive) {
File dir = new File(packagePath);
if (!dir.exists() || !dir.isDirectory()) {
return;
}
final boolean fileRecursive = recursive;
File[] dirfiles = dir.listFiles(new FileFilter() {
// 自定义文件过滤规则
public boolean accept(File file) {
if (file.isDirectory()) {
return fileRecursive;
}
String filename = file.getName();
if (excludeInner && filename.indexOf('$') != -1) {
logger.info("exclude inner class with name:" + filename);
return false;
}
return filterClassName(filename);
}
});
for (File file : dirfiles) {
if (file.isDirectory()) {
doScanPackageClassesByFile(classes, packageName + "."
+ file.getName(), file.getAbsolutePath(), recursive);
} else {
String className = file.getName().substring(0,
file.getName().length() - 6);
try {
classes.add(Thread.currentThread().getContextClassLoader()
.loadClass(packageName + '.' + className));

} catch (ClassNotFoundException e) {
logger.error("IOException error:", e);
}
}
}
}

/**
* 根据过滤规则判断类名
* @param className
* @return
*/
private boolean filterClassName(String className) {
if (!className.endsWith(".class")) {
return false;
}
if (null == this.classFilters || this.classFilters.isEmpty()) {
return true;
}
String tmpName = className.substring(0, className.length() - 6);
boolean flag = false;
for (String str : classFilters) {
String tmpreg = "^" + str.replace("*", ".*") + "$";
Pattern p = Pattern.compile(tmpreg);
if (p.matcher(tmpName).find()) {
flag = true;
break;
}
}
return (checkInOrEx && flag) || (!checkInOrEx && !flag);
}

/**
* @return the excludeInner
*/
public boolean isExcludeInner() {
return excludeInner;
}

/**
* @return the checkInOrEx
*/
public boolean isCheckInOrEx() {
return checkInOrEx;
}

/**
* @return the classFilters
*/
public List<String> getClassFilters() {
return classFilters;
}

/**
* @param pExcludeInner the excludeInner to set
*/
public void setExcludeInner(boolean pExcludeInner) {
excludeInner = pExcludeInner;
}

/**
* @param pCheckInOrEx the checkInOrEx to set
*/
public void setCheckInOrEx(boolean pCheckInOrEx) {
checkInOrEx = pCheckInOrEx;
}

/**
* @param pClassFilters the classFilters to set
*/
public void setClassFilters(List<String> pClassFilters) {
classFilters = pClassFilters;
}

/**
* @param args
*/
public static void main(String[] args) {

// 自定义过滤规则
List<String> classFilters = new ArrayList<String>();
classFilters.add("File*");

// 创建一个扫描处理器,排除内部类 扫描符合条件的类
ClassPathScanHandler handler = new ClassPathScanHandler(true, true,
classFilters);

System.out
.println("开始递归扫描jar文件的包:org.apache.commons.io 下符合自定义过滤规则的类...");
Set<Class<?>> calssList = handler.getPackageAllClasses(
"org.apache.commons.io", true);
for (Class<?> cla : calssList) {
System.out.println(cla.getName());
}
System.out.println("开始递归扫描file文件的包:michael.hessian 下符合自定义过滤规则的类...");
classFilters.clear();
classFilters.add("Hessian*");
calssList = handler.getPackageAllClasses("michael.hessian", true);
for (Class<?> cla : calssList) {
System.out.println(cla.getName());
}
}
}

测试运行结果如下:
[quote]
[color=red]开始递归扫描jar文件的包:org.apache.commons.io 下符合自定义过滤规则的类..[/color]
org.apache.commons.io.FileCleaner
org.apache.commons.io.filefilter.FileFilterUtils
org.apache.commons.io.FileUtils
[color=red]开始递归扫描file文件的包:michael.hessian 下符合自定义过滤规则的类...[/color]
michael.hessian.client.HessianClientTest
michael.hessian.client.HessianSpringClient
[/quote]

转载请注明来自:Michael's blog @[url=http://sjsky.iteye.com]http://sjsky.iteye.com[/url]

-----------------------------------分 ------------------------------------隔 ------------------------------------线 --------------------------------------
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值