`

使用classloader自定义测试套件TestSuite

    博客分类:
  • Test
阅读更多
junit自带了一个suite用来将多个test case放在一起执行, 但是有时候test case太多, 或者每次只需要对特定的几个test case进行测试, 这样写就比较繁琐, 于是希望通过一种带有通配符的表达式来指定需要测试的某些符合条件的test case, 于是根据这个需求, 实现了一个自己的PathSuite, 不过目前还比较简单, 关键是表达式的解析实现比较难搞, 不知道有没有现成的类似的通配符表达式可用? 反正我是没有找到:(
这里需要借助ClassLoader用来加载指定的class文件, 因为通过表达式得到的只是一些文件路径, 要将指定的路径转换成class定义以及类实例就需要借助ClassLoader, 当然这里使用的都是ClassLoader非常简单的功能.
/**
 * <p>
* 使用Junit Runner运行指定package下所有的*Test.java测试类或者指定的类.
* </p>
* 目前只支持形如
* <ul>
* <li>"com.mysoft.item.test.*"匹配整个package</li>
* <li>"com.mysoft.item.test.*ATest.class",匹配整个package下的某些文件(未实现)</li>
* <li>"com.mysoft.item.test.**", 匹配整个package以及子package(未实现)</li>
* <li>"com.mysoft.item.test.internal.CTest.class"匹配单个文件</li>
* </ul>
* 的表达式
 * @since 2009-10-14 下午06:00:12
 */
public class PathSuite extends CompositeRunner {
    private TestClass fTestClass;
    private final static String classSuffix = ".class";

    @Retention(RetentionPolicy.RUNTIME)
    @Target(ElementType.TYPE)
    public @interface Paths {
        /**
         * 指定需要运行的测试目录或测试类
         * @return
         */
        public String[] value();

        /**
         * 指定那些不需要运行的测试目录或测试类(未实现)
         * @return
         */
        public String[] exclude() default {};
    }

    public PathSuite(Class<?> klass) throws InitializationError {
        super(klass.getSimpleName());

        Class<?>[] annotatedClasses = getAnnotatedClasses(klass);
        for (Class<?> each : annotatedClasses) {
            Runner childRunner = Request.aClass(each).getRunner();
            if (childRunner != null)
                add(childRunner);
        }

        fTestClass = new TestClass(klass);
        MethodValidator methodValidator = new MethodValidator(fTestClass);
        methodValidator.validateStaticMethods();
        methodValidator.assertValid();
    }

    private static Class<?>[] getAnnotatedClasses(Class<?> klass) throws InitializationError {
        Paths annotation = klass.getAnnotation(Paths.class);
        if (annotation == null)
            throw new InitializationError(String.format("class '%s' must have a Paths annotation", klass
                    .getName()));

        return getTestClasses(annotation.value());
    }

    private static Class<?>[] getTestClasses(String[] testClassPaths) {
        List<String> classNames = getClassNames(testClassPaths);

        ClassLoader loader = new TestClassLoader();
        List<Class<?>> classes = new ArrayList<Class<?>>();
        for (String className : classNames) {
            try {
                Class<?> clazz = loader.loadClass(className);
                // 去掉抽象类
                if (Modifier.isAbstract(clazz.getModifiers())) {
                    continue;
                }
                classes.add(clazz);
            } catch (Exception e) {
                throw new RuntimeException(String.format("加载测试类[%s]失败", className), e);
            }
        }

        return classes.toArray(new Class<?>[0]);
    }

    private static List<String> getClassNames(String[] testClassPaths) {
        List<String> classNames = new ArrayList<String>();
        for (String testClassPath : testClassPaths) {
            // com.mysoft.item.test.*
            if (StringUtils.isBlank(testClassPath)) {
                continue;
            }
            String packagePrefix = testClassPath.replaceAll("\\*.*", "");
            if (isClass(testClassPath)) {
                // com.mysoft.item.test.ATest.class

                // com.mysoft.item.test.ATest
                testClassPath = testClassPath.replaceFirst("\\.class", "");
                // com.mysoft.item.test
                packagePrefix = testClassPath.substring(0, testClassPath.lastIndexOf(".") + 1);

                // com/mysoft/item/test/ATest.class
                testClassPath = testClassPath.replace(".", "/") + classSuffix;
            } else {
                // com.mysoft.item.test.
                packagePrefix = testClassPath.replaceAll("\\*.*", "");
                // com/mysoft/item/test/*
                testClassPath = testClassPath.replace(".", "/");
            }


            classNames.addAll(getClassNames(testClassPath, packagePrefix));
        }
        return classNames;
    }

    private static boolean isClass(String testClassPath) {
        return testClassPath.endsWith(classSuffix);
    }

    private static List<String> getClassNames(String path,
            String packagePrefix) {
        // com/mysoft/item/test
        path = path.replaceFirst("\\/\\*$", "");

        String absolutePath = PathUtils.getAbsolutePath(path);

        return findClasses(packagePrefix, new File(absolutePath));
    }

    private static List<String> findClasses(String packagePrefix, File folder) {
        List<String> result = new ArrayList<String>();
        if (folder.isDirectory()) {
            // 得到里面的每一个class文件
            File[] testClasses = folder.listFiles(new FilenameFilter() {
                public boolean accept(File dir, String name) {
                    return name.endsWith("Test.class");
                }
            });

            for (File file : testClasses) {
                result.add(getClassName(packagePrefix, file));
            }
        } else {
            result.add(getClassName(packagePrefix, folder));
        }
        return result;
    }

    private static String getClassName(String packagePrefix, File file) {
        return packagePrefix + file.getName().replaceFirst("\\.class", "");

    }
}



TestClassLoader的实现, 这个是从网上找来的一个类似的实现做了些改动, 可以凑合着用用^_^, 不过大部分ClassLoader差不多都这样用
/**
* 用来加载当前classpath下的Test Class
 * @since 2009-10-21 下午03:46:34
 */
public class TestClassLoader extends ClassLoader {

    /**
     * @param name 形如"java.lang.String"的字符串
     * {@inheritDoc} 
     */
    @Override
    protected Class<?> findClass(String name)
            throws ClassNotFoundException {
        byte[] bytes = loadClassBytes(name);
        Class<?> theClass = defineClass(name, bytes, 0, bytes.length);// A
        if (theClass == null)
            throw new ClassFormatError();
        return theClass;
    }

    private byte[] loadClassBytes(String className) throws
            ClassNotFoundException {
        try {
            String classFile = getClassFile(className);
            FileInputStream fis = new FileInputStream(classFile);
            FileChannel fileC = fis.getChannel();
            ByteArrayOutputStream baos = new ByteArrayOutputStream();
            WritableByteChannel outC = Channels.newChannel(baos);
            ByteBuffer buffer = ByteBuffer.allocateDirect(1024);
            while (true) {
                int i = fileC.read(buffer);
                if (i == 0 || i == -1) {
                    break;
                }
                buffer.flip();
                outC.write(buffer);
                buffer.clear();
            }
            fis.close();
            return baos.toByteArray();
        } catch (IOException fnfe) {
            throw new ClassNotFoundException(className);
        }
    }

    private String getClassFile(String name) {
        StringBuffer sb = new StringBuffer(getBaseDir());
        name = name.replace('.', File.separatorChar) + ".class";
        sb.append(File.separator + name);
        return sb.toString();
    }

    private String getBaseDir() {
        return TestClassLoader.class.getResource(File.pathSeparator).toString();
    }

    // 新增的一个findResource方法
    @Override
    protected URL findResource(String name) {
        try {
            URL url = super.findResource(name);
            if (url != null)
                return url;
            url = new URL("file:///" + converName(name));
            // 简化处理,所有资源从文件系统中获取
            return url;
        } catch (MalformedURLException mue) {
            throw new RuntimeException(mue);
        }
    }

    private String converName(String name) {
        StringBuffer sb = new StringBuffer(getBaseDir());
        name = name.replace('.', File.separatorChar);
        sb.append(File.separator + name);
        return sb.toString();
    }
}



这里只是为了演示classloader的用法, 其实根本没有必要这么复杂, 其中
Class<?> clazz = loader.loadClass(className);

这一句, 可以简单的使用
Class<?> clazz = Class.forName(className);

代替搞定.
另外eclipse也可以直接选中指定的package运行里面所有的test case, 我也是最近从测试人员那里知晓的:(
0
0
分享到:
评论
1 楼 HashSet 2014-03-25  
另外eclipse也可以直接选中指定的package运行里面所有的test case, 我也是最近从测试人员那里知晓的:(


怎么做

相关推荐

    基于微信小程序的在线办公小程序答辩PPT.pptx

    基于微信小程序的在线办公小程序答辩PPT.pptx

    机器学习(预测模型):2000年至2015年期间193个国家的预期寿命和相关健康因素的数据

    这个数据集来自世界卫生组织(WHO),包含了2000年至2015年期间193个国家的预期寿命和相关健康因素的数据。它提供了一个全面的视角,用于分析影响全球人口预期寿命的多种因素。数据集涵盖了从婴儿死亡率、GDP、BMI到免疫接种覆盖率等多个维度,为研究者提供了丰富的信息来探索和预测预期寿命。 该数据集的特点在于其跨国家的比较性,使得研究者能够识别出不同国家之间预期寿命的差异,并分析这些差异背后的原因。数据集包含22个特征列和2938行数据,涉及的变量被分为几个大类:免疫相关因素、死亡因素、经济因素和社会因素。这些数据不仅有助于了解全球健康趋势,还可以辅助制定公共卫生政策和社会福利计划。 数据集的处理包括对缺失值的处理、数据类型转换以及去重等步骤,以确保数据的准确性和可靠性。研究者可以使用这个数据集来探索如教育、健康习惯、生活方式等因素如何影响人们的寿命,以及不同国家的经济发展水平如何与预期寿命相关联。此外,数据集还可以用于预测模型的构建,通过回归分析等统计方法来预测预期寿命。 总的来说,这个数据集是研究全球健康和预期寿命变化的宝贵资源,它不仅提供了历史数据,还为未来的研究和政策制

    基于微信小程序的“健康早知道”微信小程序答辩PPT.pptx

    基于微信小程序的“健康早知道”微信小程序答辩PPT.pptx

    基于微信小程序的电影交流平台答辩PPT.pptx

    基于微信小程序的电影交流平台答辩PPT.pptx

    计算机字符编码GB18030.PDF

    计算机字符编码GB18030

    Hive 操作基础(进阶版)多级分区数据文件2

    Hive 操作基础(进阶版)多级分区数据文件2

    基于java的贫困生管理系统答辩PPT.pptx

    基于java的贫困生管理系统答辩PPT.pptx

    pandas-2.1.4-cp312-cp312-win_amd64.zip

    pandas whl安装包,对应各个python版本和系统(具体看资源名字),找准自己对应的下载即可! 下载后解压出来是已.whl为后缀的安装包,进入终端,直接pip install pandas-xxx.whl即可,非常方便。 再也不用担心pip联网下载网络超时,各种安装不成功的问题。

    TA_Lib轮子无需编译-TA_Lib-0.4.18-cp38-cp38-win32.whl.zip

    TA_lib库(whl轮子),直接pip install安装即可,下载即用,非常方便,各个python版本对应的都有。 使用方法: 1、下载下来解压; 2、确保有python环境,命令行进入终端,cd到whl存放的目录,直接输入pip install TA_lib-xxxx.whl就可以安装,等待安装成功,即可使用! 优点:无需C++环境编译,下载即用,方便

    课设毕设基于SpringBoot+Vue的瑜伽体验课预约系统源码可运行.zip

    本压缩包资源说明,你现在往下拉可以看到压缩包内容目录 我是批量上传的基于SpringBoot+Vue的项目,所以描述都一样;有源码有数据库脚本,系统都是测试过可运行的,看文件名即可区分项目~ |Java|SpringBoot|Vue|前后端分离| 开发语言:Java 框架:SpringBoot,Vue JDK版本:JDK1.8 数据库:MySQL 5.7+(推荐5.7,8.0也可以) 数据库工具:Navicat 开发软件: idea/eclipse(推荐idea) Maven包:Maven3.3.9+ 系统环境:Windows/Mac

    tornado-6.2b2.tar.gz

    tornado-6.2b2.tar.gz

    javawe论坛项目 原生技术

    javawe论坛项目 原生技术

    tornado-6.2b1-cp310-cp310-macosx_10_9_universal2.whl

    tornado-6.2b1-cp310-cp310-macosx_10_9_universal2.whl

    基于司机信用评价的货运管理系统(springboot+vue+mysql+说明文档).zip

    随着物流行业的快速发展,货运管理变得愈发重要。为了提高货运效率,确保货物安全,我们开发了这款基于司机信用评价的货运管理系统。 该系统主要包含了货物信息管理、订单评价管理、货主管理等多个功能模块。在货物信息管理模块中,用户可以查看和管理货物的详细信息,包括货物名称、规格、装车状态、运输状态以及卸货状态等,方便用户随时掌握货物的动态。 订单评价管理模块是该系统的核心之一,它允许货主对司机的服务进行评价,系统会根据评价数据对司机进行信用评分。这一功能不仅有助于提升司机的服务质量,还能为货主提供更加可靠的货运选择。 此外,货主管理模块提供了货主信息的录入、修改和查询等功能,方便用户管理自己的货主资料。系统界面简洁明了,以蓝色为主色调,设计现代且专业,为用户提供了良好的使用体验。 通过该系统,用户可以轻松实现货物信息的查看和管理,对司机的服务进行评价,提高货运效率和服务质量。同时,系统也为司机提供了一个展示自我、提升信用的平台,有助于推动物流行业的健康发展。

    毕业生交流学习平台 SSM毕业设计 附带论文.zip

    毕业生交流学习平台 SSM毕业设计 附带论文 启动教程:https://www.bilibili.com/video/BV1GK1iYyE2B

    基于java的广场舞团答辩PPT.pptx

    基于java的广场舞团答辩PPT.pptx

    基于java的基于SSM的校园音乐平台答辩PPT.pptx

    基于java的基于SSM的校园音乐平台答辩PPT.pptx

    安装包JIRATimeSLA

    Jira插件安装包

    【java毕业设计】基于图像识别与分类的中国蛇类识别系统源码(springboot+vue+mysql+说明文档).zip

    项目经过测试均可完美运行! 环境说明: 开发语言:java jdk:jdk1.8 数据库:mysql 5.7+ 数据库工具:Navicat11+ 管理工具:maven 开发工具:idea/eclipse

    tornado-6.2b2-cp37-abi3-win_amd64.whl

    tornado-6.2b2-cp37-abi3-win_amd64.whl

Global site tag (gtag.js) - Google Analytics