`
sjsky
  • 浏览: 916765 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

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

    博客分类:
  • Java
阅读更多
    blog迁移至:http://www.micmiu.com

本文永久链接:http://sjsky.iteye.com/blog/1061092

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

一、方法说明
// 无参构造器,默认是排除内部类、并搜索符合规则
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);


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


三、处理类代码
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());
        }
    }
}

测试运行结果如下:
引用

开始递归扫描jar文件的包:org.apache.commons.io 下符合自定义过滤规则的类..
org.apache.commons.io.FileCleaner
org.apache.commons.io.filefilter.FileFilterUtils
org.apache.commons.io.FileUtils
开始递归扫描file文件的包:michael.hessian 下符合自定义过滤规则的类...
michael.hessian.client.HessianClientTest
michael.hessian.client.HessianSpringClient


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

-----------------------------------分 ------------------------------------隔 ------------------------------------线 --------------------------------------
3
1
分享到:
评论
1 楼 yarafa 2013-06-13  
哥们,你的程序放到JBoss 5.0下就不灵了,请问在JBoss下怎么处理,目前正在纠结这个问题,尤其是JAR包的扫描

相关推荐

    jar包冲突检测工具.zip

    它的工作原理可能是通过遍历指定路径下的所有jar文件,然后对比每个jar中的类文件,找出具有相同全限定名(即包括包名和类名)但内容不同的类,从而确定冲突的存在。 `CallJars.exe` 可能是该工具的执行文件,它是...

    spring 扫描jar maven 打包

    Spring通过`@Component`、`@Service`、`@Repository`和`@Controller`等注解标记类,然后使用`@ComponentScan`注解来指定哪些包应该被扫描,以发现这些标记的bean。 ```java @Configuration @ComponentScan...

    Spring EntityManager 不能扫描jar 中的class文件

    这是因为Spring的`ClassPathScanningCandidateComponentProvider`类主要针对类路径下的目录和文件进行扫描,而不是直接处理jar文件。 为了解决这个问题,我们需要采取以下几种策略: 1. **显式配置**:在Spring...

    Java服务器Tld标签文件扫描器(含源代码)

    J2EE服务器启动时,会扫描每个jar文件,检查其是否包含标签定义文件,及.tld后缀的文件,随着项目规模扩大,用到的jar包文件越来越多,检查每个jar文件会延缓服务器启动时间,比如Tomcat启动时如果检测到扫描了不含....

    Java的包扫描实现和Jar应用.docx

    包扫描是一种机制,它允许我们在运行时查找并处理指定包或其子包下的所有类。在Spring框架中,这种机制被广泛用于自动装配和组件扫描,以便在启动时识别和初始化相关的bean。在自定义的项目中,我们也可以利用这个...

    用于二维码扫描的jar包

    通常,这样的zip文件会包含一个或多个jar文件,这些文件包含了二维码生成和扫描所需的类和方法。 在Java中,常用的二维码处理库有ZXing(Zebra Crossing)和QRGen等。ZXing是一个开源项目,提供了多种条码和二维码...

    上传文件使用的jar包jspsmartupload.jar

    jar包是Java平台上的一个归档文件格式,用于集合多个Java类文件、相关的元数据以及资源文件,便于分发和部署。在本例中,"jspsmartupload.jar"包含了实现文件上传功能所需的全部类和资源,开发者可以直接引入项目中...

    用Fat Jar Eclipse Plug-In打包可执行jar文件

    Fat Jar插件是Eclipse的一个扩展,它允许开发者轻松地将所有必需的类和资源,包括外部库,打包进一个单一的、可执行的jar文件中。使用这个插件,开发者可以避免手动管理类路径,减少了出错的可能性。 具体操作步骤...

    jar包-JavaWeb所用到的jar包

    JAR文件本质上是一个ZIP格式的压缩文件,包含了Java应用程序或库的所有组件,使得开发者能够更有效地管理和使用Java资源。以下是一些关于Java Web开发中JAR包的重要知识点: 1. **JAR文件结构**: - `META-INF`...

    将jar包打包成一个exe执行文件

    - 打包时要确保JAR文件无误,且所有依赖库都被包含在内。 - 如果使用私有JRE,确保其与你的代码兼容,且符合目标用户的操作系统需求。 - 生成的exe文件可能受反病毒软件误报,因此在分发前最好进行病毒扫描。 以上...

    mybatis自动生成文件jar包

    "mybatis自动生成文件jar包"是指包含MyBatis代码生成器相关功能的JAR文件。这个JAR包通常包含了MyBatis代码生成器的类和依赖,使得用户在不安装完整MyBatis框架的情况下,也能运行代码生成器,生成所需的Java代码。 ...

    打包jar文件Eclipse插件FatJar安装与使用.pdf

    然而,在某些场景下,我们可能希望将项目的依赖库、配置文件等一并包含在一个单一的`.jar`文件中,以减少部署的复杂度。这时,Eclipse的FatJar插件便派上了用场。 #### 二、FatJar插件简介 FatJar插件是为了解决...

    fatjar打包工具

    fatjar的工作原理是通过扫描项目的类路径,包括源代码、资源文件以及所有依赖库,然后将它们打包到一个独立的JAR文件中。这样,当你运行这个JAR时,所有必要的类和资源都会包含在内,避免了运行时因找不到依赖而产生...

    Ubuntu下make jar包

    这里的`fileTree`会扫描指定目录下的所有jar文件,并将其作为依赖项引入。确保第三方jar包已经放在项目的`libs`目录下。 然后,执行`./gradlew build`命令(或者在Android Studio中点击Build &gt; Rebuild Project)来...

    XZbar.jar 以及扫描框资源文件

    在使用时,只需将此jar文件导入到项目类路径中,通过相应的API调用即可实现扫描和解码功能。 接下来,我们关注一下随XZbar.jar一起提供的资源文件。这些文件在实现扫描界面和交互体验中起着至关重要的作用: 1. ...

    Eclipse打包工具FATJAR

    4. **完成打包**:最后,指定输出JAR文件的位置,点击Finish,Eclipse就会生成一个包含了所有依赖的单个可执行JAR文件。 **注意事项** - FATJAR打包过程中可能会遇到类重复或资源文件覆盖的问题,此时需要检查项目...

    fatjar_0.0.27将java打包成jar文件的工具及演示视频

    这样的JAR文件包含了运行程序所需的所有资源,使得在没有完整构建环境的情况下也能直接运行。`FatJar_0.0.27`是该工具的一个版本,通过它,开发者可以简化打包过程,避免处理复杂的类路径和依赖管理问题。 `FatJar`...

    Spring自动扫描无法扫描jar包中bean的解决方法

    不勾选"Add directory entries"时,jar包内部的类文件会按照扁平化的结构存储,这样Spring在扫描时无法定位到类的正确位置,因此无法加载Bean。而勾选这个选项后,jar包内的类文件会被组织成具有完整目录结构的形式...

    Mybatis_Spring整合jar包

    - 配置MapperScannerConfigurer,自动扫描指定包下的Mapper接口,Spring会为每个接口生成一个代理实现类。 - 在Mapper接口中定义SQL操作,MyBatis的XML配置文件中编写具体的SQL语句。 - 使用@Autowired注解,让...

    Spring_mybatis整合相关 jar 包,配置文件

    3. 配置MapperScannerConfigurer:扫描指定包下的所有Mapper接口,自动注册为Spring的bean。 4. 配置事务管理器:一般使用`PlatformTransactionManager`,并配置事务的隔离级别、传播行为等。 5. 使用@Autowired或...

Global site tag (gtag.js) - Google Analytics