类路径扫描
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
spring启动componentscan类扫描加载过程—源码分析Java开发Java经验技巧共16页.pdf.zip
湖南省益阳市大通湖区2017年中考化学专题练测2物质的分类扫描版无答案
扫描全能王提供高品质、最专业、最便捷的扫描、传真服务,将您的智能手机变为随身携带的扫描仪、传真机
本软件禁止用于黑客用途,不允许对除个人网站外进行扫描检测,不允许利用漏洞进行入侵操作,若您使用本程序进行违法操作,造成的任何后果与作者无关,作者不承担任何法律责任、纠纷,软件作为一款安全检测软件,...
类扫描器流 这是一个轻量级的类扫描器,它使我们可以在流管道中应用过滤器并收集结果。 扫描指定包下的所有类 List< Class> scanResults = ClassScanner . scan(classLoader, " com.beerboy.scanner " ) .collect...
1.开发机是Debian,在Windows上跑分隔符没考虑 2.Ide环境下没问题,但是在打成jar包部署的时候出问题 3.不同的部署环境下又有问题,比如打成s
为简单起见,本程序只处理了开发环境自动生成的窗体类定义文件。 本程序中用到的SkyEdit、SkyParser控件是我去年底完成的彩色语法显示编辑器控件(地址:http://www.2ccc.com/article.asp?articleid=1764),你可在本...
可以显示多张图片,对显示的图片放大,缩小,显示高级
通常 情况下是按照三维激光扫描仪有效的扫描距离来进行分类,可分为: (1)短距离激光扫描仪:通常情况下最佳的扫描距离为0.6~1.2 m,最长扫描距离也不超过3m,这类扫描仪通常适合用于小型的模具的量测,不仅扫描...
C#开发:COM口通讯类扫描枪,接受扫描枪扫描的信息 程序中包含Skins一些皮肤的类供参考
wifi工具类优化wifi工具类扫描结果 2018.1.25 添加下载模块 支持断点下载,多任务下载 删除下载 添加下载模块测试用例 2018.2.20 优化工具类调用 wifi工具类和蓝牙工具类优化 实现单例 添加打字机效果 ...
直接就可以用来扫描,和微信类似,可以进行二维码和条形码的扫描,扫描之后显示网址
Tag v2.0受b0uya师傅帮助,了解了Python3下的asyncio协程高并发,配合aiohttp比之前快了三倍有余,不考虑服务器压力影响下,可以与御剑这类扫描工具抗衡。 由于 Windows 下拥有众多优秀目录扫描工具,且速度很快;...
进行秘密SYN扫描,对象为主机Saznme所在的“C类”网段 的255台主机。同时尝试确定每台工作主机的操作系统类型。因为进行SYN扫描 和操作系统检测,这个扫描需要有根权限。 nmap -sV -p 22,53,110,143,4564 ...
Android版扫描枪,清查终端设备,使用广播方式,通过代码对扫描枪设备自带扫描键进行监听。当用户按扫描枪上的扫描键时触发扫描事件,扫描后获取到扫描内容。现在没有办法免费了,只能设置1分了。
类C语言的词法扫描器,通过设计调试词法分析程序,实现从源程序中分出各种单词的方法;加深对课堂教学的理解;提高词法分析方法的实践能力
掌握标准C++ 类 .扫描版 掌握标准C++ 类 .扫描版 掌握标准C++ 类 .扫描版
c#窗口扫描类控件,枚举顶层窗口,列举出窗口的绝大部分信息,包括窗口句柄,客户区,窗口区,标题,类名,截图等等 --游荡男孩 www.boitboy.com
扫描设计是一种广泛采用的可测性设计方法。在采用扫描设计的电路中,扫描单元及其控制电路芯片面积可能占到30%,引起的故障总数可能占到50%。因此扫描链的诊断对于逻辑诊断具有重要的意义。本文综述了在采用扫描结构...