`
yanguz123
  • 浏览: 556908 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
社区版块
存档分类
最新评论

类扫描

    博客分类:
  • Code
 
阅读更多

 

 

类路径扫描

package com.yuan.common.annotation;

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

/**
 * 类路径扫描
 * 
 */
public class ClassPathScanner {

	public static void main(String[] args) {
		try {
			Set<Class<?>> classes = new ClassPathScanner().getPackageAllClasses("com.yuan", true);
			for (Class<?> clazz : classes) {
				System.out.println(clazz.getName());
			}
		} catch (ClassNotFoundException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} catch (IOException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
	}

	// 是否包括内部类
	private boolean excludeInner = true;
	private boolean checkInOrEx = true;
	// 根据类名过滤
	private List<String> classFilters = null;

	public ClassPathScanner() {
	}

	public ClassPathScanner(Boolean excludeInner, Boolean checkInOrEx, List<String> classFilters) {
		this.excludeInner = excludeInner;
		this.checkInOrEx = checkInOrEx;
		this.classFilters = classFilters;
	}

	public boolean isExcludeInner() {
		return excludeInner;
	}

	public void setExcludeInner(boolean excludeInner) {
		this.excludeInner = excludeInner;
	}

	public boolean isCheckInOrEx() {
		return checkInOrEx;
	}

	public void setCheckInOrEx(boolean checkInOrEx) {
		this.checkInOrEx = checkInOrEx;
	}

	public List<String> getClassFilters() {
		return classFilters;
	}

	public void setClassFilters(List<String> classFilters) {
		this.classFilters = classFilters;
	}

	/**
	 * 获取包中的所有类
	 * 
	 * @param basePackage
	 * @param recursive
	 * @return
	 * @throws IOException
	 * @throws ClassNotFoundException
	 */
	public Set<Class<?>> getPackageAllClasses(String basePackage, boolean recursive) throws IOException, ClassNotFoundException {
		// 使用LinkedHashSet来存放扫描到的类
		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 = Thread.currentThread().getContextClassLoader().getResources(package2Path);
		while (dirs.hasMoreElements()) {
			URL url = dirs.nextElement();
			// 获取URL协议
			String protocol = url.getProtocol();
			if ("file".equals(protocol)) {
				String filePath = URLDecoder.decode(url.getFile(), "UTF-8");
				// 扫描文件夹中的包和类
				doScanPackageClassesByFile(classes, packageName, filePath, recursive);
			} else if ("jar".equals(protocol)) {
				// 扫描jar包中的包和类
				doScanPackageClassesByJar(packageName, url, recursive, classes);
			}
		}
		return classes;

	}

	/**
	 * 在jar包中扫描包和类
	 * 
	 * @param basePackage
	 *            包名
	 * @param url
	 *            类路径
	 * @param recursive
	 *            是否递归
	 * @param classes
	 *            传引用,返回的结果
	 * @throws IOException
	 * @throws ClassNotFoundException
	 */
	private void doScanPackageClassesByJar(String basePackage, URL url, final boolean recursive, Set<Class<?>> classes) throws IOException, ClassNotFoundException {
		// 包名
		String packageName = basePackage;
		// 获取文件路径
		String package2Path = 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(package2Path) || entry.isDirectory()) {
				continue;
			}
			// 判断是否递归搜索子包
			if (!recursive && name.lastIndexOf('/') != package2Path.length()) {
				continue;
			}
			// 判断是否过滤 inner class
			if (this.excludeInner && name.indexOf('$') != -1) {
				continue;
			}
			String classSimpleName = name.substring(name.lastIndexOf('/') + 1);
			// 判定是否符合过滤条件
			if (this.filterClassName(classSimpleName)) {
				String className = name.replace('/', '.');
				className = className.substring(0, className.length() - 6);
				// 用当前线程的类加载器加载类
				classes.add(Thread.currentThread().getContextClassLoader().loadClass(className));
			}
		}
	}

	/**
	 * 在文件夹中扫描包和类
	 * 
	 * @param classes
	 * @param packageName
	 * @param packagePath
	 * @param recursive
	 * @throws ClassNotFoundException
	 */
	private void doScanPackageClassesByFile(Set<Class<?>> classes, String packageName, String packagePath, boolean recursive) throws ClassNotFoundException {
		// 转为文件
		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) {
					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);
				classes.add(Thread.currentThread().getContextClassLoader().loadClass(packageName + '.' + className));
			}
		}
	}

	/**
	 * 过滤类文件
	 * 
	 * @param className
	 * @return
	 */
	private boolean filterClassName(String className) {
		// 文件后缀为class
		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);
	}

}

 

 

分享到:
评论

相关推荐

    PHPmysql数据库操作类扫描.pdf

    PHPmysql数据库操作类扫描.pdf

    spring启动componentscan类扫描加载过程

    spring启动componentscan类扫描加载过程—源码分析Java开发Java经验技巧共16页.pdf.zip

    湖南省益阳市大通湖区2017年中考化学专题练测2物质的分类扫描版无答案

    湖南省益阳市大通湖区2017年中考化学专题练测2物质的分类扫描版无答案

    扫描全能王

    扫描全能王提供高品质、最专业、最便捷的扫描、传真服务,将您的智能手机变为随身携带的扫描仪、传真机

    北极熊扫描器3.5【国产安全工具】

    本软件禁止用于黑客用途,不允许对除个人网站外进行扫描检测,不允许利用漏洞进行入侵操作,若您使用本程序进行违法操作,造成的任何后果与作者无关,作者不承担任何法律责任、纠纷,软件作为一款安全检测软件,...

    class-scanner-stream:类扫描器作为流

    类扫描器流 这是一个轻量级的类扫描器,它使我们可以在流管道中应用过滤器并收集结果。 扫描指定包下的所有类 List&lt; Class&gt; scanResults = ClassScanner . scan(classLoader, " com.beerboy.scanner " ) .collect...

    iSenninha#jianShuRecord#mybatis的类扫描工具1

    1.开发机是Debian,在Windows上跑分隔符没考虑 2.Ide环境下没问题,但是在打成jar包部署的时候出问题 3.不同的部署环境下又有问题,比如打成s

    Form Class Scaner (BCB/Delphi窗体类扫描机)

    为简单起见,本程序只处理了开发环境自动生成的窗体类定义文件。 本程序中用到的SkyEdit、SkyParser控件是我去年底完成的彩色语法显示编辑器控件(地址:http://www.2ccc.com/article.asp?articleid=1764),你可在本...

    地籍档案扫描显示

    可以显示多张图片,对显示的图片放大,缩小,显示高级

    毕业设计-基于地面三维激光扫描技术的三维模型重建.doc

    通常 情况下是按照三维激光扫描仪有效的扫描距离来进行分类,可分为: (1)短距离激光扫描仪:通常情况下最佳的扫描距离为0.6~1.2 m,最长扫描距离也不超过3m,这类扫描仪通常适合用于小型的模具的量测,不仅扫描...

    COM口通讯扫描枪程序

    C#开发:COM口通讯类扫描枪,接受扫描枪扫描的信息 程序中包含Skins一些皮肤的类供参考

    android 开发常用工具类集合

    wifi工具类优化wifi工具类扫描结果 2018.1.25 添加下载模块 支持断点下载,多任务下载 删除下载 添加下载模块测试用例 2018.2.20 优化工具类调用 wifi工具类和蓝牙工具类优化 实现单例 添加打字机效果 ...

    Zxing扫描工具类

    直接就可以用来扫描,和微信类似,可以进行二维码和条形码的扫描,扫描之后显示网址

    基于Python的网站路径扫描工具+源代码+文档说明

    Tag v2.0受b0uya师傅帮助,了解了Python3下的asyncio协程高并发,配合aiohttp比之前快了三倍有余,不考虑服务器压力影响下,可以与御剑这类扫描工具抗衡。 由于 Windows 下拥有众多优秀目录扫描工具,且速度很快;...

    nmap 扫描端口使用

     进行秘密SYN扫描,对象为主机Saznme所在的“C类”网段 的255台主机。同时尝试确定每台工作主机的操作系统类型。因为进行SYN扫描 和操作系统检测,这个扫描需要有根权限。  nmap -sV -p 22,53,110,143,4564 ...

    Android监听扫描枪设备自带扫描键DemoT

    Android版扫描枪,清查终端设备,使用广播方式,通过代码对扫描枪设备自带扫描键进行监听。当用户按扫描枪上的扫描键时触发扫描事件,扫描后获取到扫描内容。现在没有办法免费了,只能设置1分了。

    类C语言的词法扫描器

    类C语言的词法扫描器,通过设计调试词法分析程序,实现从源程序中分出各种单词的方法;加深对课堂教学的理解;提高词法分析方法的实践能力

    掌握标准C++ 类 .扫描版

    掌握标准C++ 类 .扫描版 掌握标准C++ 类 .扫描版 掌握标准C++ 类 .扫描版

    c# 窗口扫描类控件

    c#窗口扫描类控件,枚举顶层窗口,列举出窗口的绝大部分信息,包括窗口句柄,客户区,窗口区,标题,类名,截图等等 --游荡男孩 www.boitboy.com

    集成电路扫描链诊断技术

    扫描设计是一种广泛采用的可测性设计方法。在采用扫描设计的电路中,扫描单元及其控制电路芯片面积可能占到30%,引起的故障总数可能占到50%。因此扫描链的诊断对于逻辑诊断具有重要的意义。本文综述了在采用扫描结构...

Global site tag (gtag.js) - Google Analytics