ANTLR和StringTemplate实例:自动生成单元测试类

ANTLR和StringTemplate实例:自动生成单元测试类

1. ANTLR语法

要想自动生成单元测试,首先第一步就是分析被测试类。这里以Java代码为例,用ANTLR对Java代码进行分析。要想靠自己完全手写出一门语言的ANTLR语法文件的复杂程度难以想象,很贴心的是在ANTLR的GitHub网站上列出了很多常见语言的语法文件,例如Java,Sqlite和MySQL的SQL语法等。

有了.g4语法文件,按照Antlr v4入门教程和实例中的步骤,就能自动生成出解析器的代码,这里就不再详述了。

2. StringTemplate基础

StringTemplate(简称ST)也是ANTLR提供的一个非常好用的工具。它的功能类似于Velocity、FreeMaker等模板引擎,可以根据事先定义好的模板,在运行时根据不同的传值渲染出不同的网页、邮件、代码等。但从名称中的String就能看出,它是比较轻量级的。试用了一下的确如此,支持在Java中硬编码简单的模板。

但试用过程中还是碰到了有不少问题,总感觉它的模板语法有些复杂啊!而且最新的ST4的API与之前版本发生了很大变化,网上找的很多例子都不好用了。具体还是参考官网的教程吧,以及这个CheatSheet表格。当然,还有花了不少时间调试好的本文的代码示例!

3. 单元测试生成器

首先来看Main方法。输入文本就是code变量表示的一个Java类,里面有两个方法。然后使用自动生成出的ANTLR代码,构建起语义分析器和解析器的处理链,并传入UnitTestGenerator监听器对输入文本进行遍历。

public class JavaCodeParseTest {

    public static void main(String[] args) {
        String code =
                "package com.jcache.store;" +
                "public class CacheStore {" +
                    "Object getCache(int a) {" +
                        "if (a == 1)" +
                            "return 1;" +
                        "else " +
                            "return 2;" +
                    "}" +
                    "void setCache(int a) {" +
                        "return;" +
                    "}" +
                "}";

        // 1.Lexical analysis
        JavaLexer lexer = new JavaLexer(new ANTLRInputStream(code));
        CommonTokenStream tokens = new CommonTokenStream(lexer);

        // 2.Syntax analysis
        JavaParser parser = new JavaParser(tokens);
        ParseTree tree = parser.compilationUnit();

        // 3.Application based on Syntax Tree
        ParseTreeWalker walker = new ParseTreeWalker();
        walker.walk(new UnitTestGenerator(), tree);
    }

}

下面就来看一个核心代码UnitTestGenerator的实现。

这里简单说一下代码中的几个关键点:

  • ST模板组:在模板文件中定义比较简单,如果要在Java中定义模板的话,一定要参照本例static初始化块的写法。
  • 嵌套子模板和Multi-valued:要想根据方法名列表,自动为每个方法名都生成一个方法,就需要按照
/**
 * Simple unit test generator.
 */
public class UnitTestGenerator extends JavaBaseListener {

    /** Constants: template name, placeholder name, generated code name */
    private static final String CLASS_ST_NAME = "classST";
    private static final String METHOD_ST_NAME = "methodST";

    private static final String TEST_PKG_NAME = "testPkgName";
    private static final String TEST_CLASS_NAME = "testClassName";
    private static final String TEST_METHOD_NAME = "testMethodName";

    private static final String CLASS_NAME_SUFFIX = "Test";
    private static final String METHOD_NAME_PREFIX = "test";

    /** Template for Java */
    private static final String METHOD_ST =
            t("@Test") +
            t("public void " + $(TEST_METHOD_NAME) + "() throws Exception {") +
            tt("//body...") +
            t("}") +
            n("");

    private static final String CLASS_ST =
            n("package " + $(TEST_PKG_NAME) + ";") +
            n("") +
            n("import org.junit.*;") +
            n("") +
            n("public class " + $(TEST_CLASS_NAME) + " {") +
            n("") +
            /**
             * Apply nested template ‘methodST‘ to multi-valued attributes ‘testMethodName‘.
             * NOTE: <attribute:template(argument-list)>
             *      Apply template to attribute with optional argument-list.
             *      Example: <name:bold()> applies bold() to name‘s value.
             *      The first argument of the template gets the iterated value.
             */
            n($(TEST_METHOD_NAME + ":" + METHOD_ST_NAME +"();separator=\"\n\"")) +
            n("}");

    /** ST group to nest template */
    private static STGroup group;
    static {
        group = new STGroup(‘$‘, ‘$‘);

        CompiledST classST = group.defineTemplate(CLASS_ST_NAME, CLASS_ST);
        classST.addArg(new FormalArgument(TEST_PKG_NAME));
        classST.addArg(new FormalArgument(TEST_CLASS_NAME));
        classST.addArg(new FormalArgument(TEST_METHOD_NAME));

        CompiledST methodST = group.defineTemplate(METHOD_ST_NAME, METHOD_ST);
        methodST.addArg(new FormalArgument(TEST_METHOD_NAME));
    }

    /** Attributes. NOTE: group.getInstanceOf() return new ST */
    private Map<String, Object> attributeMap = new HashMap<>();

    @Override
    public void enterPackageDeclaration(@NotNull JavaParser.PackageDeclarationContext ctx) {
        attributeMap.put(TEST_PKG_NAME, ctx.qualifiedName().getText());
    }

    @Override
    public void enterClassDeclaration(@NotNull ClassDeclarationContext ctx) {
        String orgClassName = ctx.Identifier().getText();
        String testClassName = orgClassName + CLASS_NAME_SUFFIX;

        attributeMap.put(TEST_CLASS_NAME, testClassName);
    }

    @Override
    public void enterMethodDeclaration(@NotNull MethodDeclarationContext ctx) {
        String orgMethodName = ctx.Identifier().getText();
        String testMethodName = METHOD_NAME_PREFIX + orgMethodName.substring(0, 1).toUpperCase()
                + orgMethodName.substring(1);

        // Multi-valued attribute
        List<String> methodNames = (List<String>) attributeMap.get(TEST_METHOD_NAME);
        if (methodNames == null) {
            methodNames = new ArrayList<>();
            attributeMap.put(TEST_METHOD_NAME, methodNames);
        }
        methodNames.add(testMethodName);
    }

    @Override
    public void enterStatement(@NotNull StatementContext ctx) {
        // If/else/switch, for/while, try-catch
        switch (ctx.getStart().getText()) {
            case "if":
                break;
            case "for":
                break;
            case "try":
                break;
            default:
                break;
        }
    }

    @Override
    public void exitCompilationUnit(@NotNull JavaParser.CompilationUnitContext ctx) {
        ST template = group.getInstanceOf(CLASS_ST_NAME);
        attributeMap.entrySet().forEach(e -> template.add(e.getKey(), e.getValue()));
        System.out.println(template.render());

        // Cleanup
        attributeMap.clear();
    }

    // =======================================
    //          String Utility
    // =======================================

    private static String $(String attrName) {
        return "$" + attrName + "$";
    }

    private static String n(String str) {
        return str + "\n";
    }

    private static String t(String str) {
        return "\t" + n(str);
    }

    private static String tt(String str) {
        return "\t\t" + n(str);
    }

}
时间: 2024-11-20 12:00:33

ANTLR和StringTemplate实例:自动生成单元测试类的相关文章

mybatis怎样自动生成java类,配置文件?

其实没有什么东西是可以自动生成的,只不过是别人已经写好了,你调用罢了. 所以想要mybatis自动生成java类,配置文件等,就必须要一些配置和一些jar包.当然这些配置也很简单. 为了有个初步的认识,首先我列出了所需要的文件: 其中标红的比较重要.好了,让我们开始吧 1.首先需要在数据库建好表,随便建几个就好. 2.下载mybatis-generator-core包 下载地址:http://search.maven.org/ 然后搜索mybatis-generator-core下载即可 3.下

【原创】有关Silverlight中自动生成的类中 没有WCF层edmx模型新加入的对象 原因分析。

前端页面层: 编译老是不通过,报如下如所示错误: -- 然后下意识的查了下 生成的cs文件,没有搜到根据edmx 生成的 对应的类. 结果整理: 1.尽管在 edmx 模型中加入了 对应的表,但 如果在 wcf层是 没有 显示的去 写方法 去调用的话, silverlight 自动生成的 类 里面 也是不会 出现该类的! 解决措施: 在WCF层的一些Service类中显示的 用下 目标对象.

Mybatis自动生成实体类、dao接口和mapping映射文件

由于Mybatis是一种半自动的ORM框架,它的工作主要是配置mapping映射文件,为了减少手动书写映射文件,可以利用mybatis生成器,自动生成实体类.dao接口以及它的映射文件,然后直接拷贝到工程中稍微修改就可以直接使用了. 生成器目录如下: 首先进入lib文件夹中,该目录如下: (图上文件下载地址:http://download.csdn.net/detail/qiwei31229/9790909) 主要修改generatorConfig.xml <?xml version="1

根据Json字符串自动生成model类(java)

根据Json自动生成Model类 java 将json转换成java类 http://jsongen.byingtondesign.com you bring the json, we'll bring the code 发现一个很不错的网站,能够直接将json字符串生成java model类,超级方便. 很久之前就知道,一直没有用过,昨天在弄G+的数据的时候,用上了. 再结合Gson,你基本上不用做些什么了,很快就能转换成java对象了. 用法:找一个数据很全的json大数据,保存为.json

C#集合篇,在业务背景下(***产品升级管理):依赖注入,变量声明,三元表达式,常用字符串相关操作方法,ADO.NET,EF机制,T4模板自动生成实体类,ref变量巧用,属性实际运用,唯一性验证

QQ:1187362408 欢迎技术交流和学习 关于系统产品升级报告管理,业务需求: TODO: 1,升级报告管理:依据各县区制定升级报告(关联sAreaCode,给每个地区观看具体升级报告信息) 2,运用的技术:依赖注入,变量声明,三元表达式,常用字符串相关操作方法,ADO.NET,EF机制,T4模板自动生成实体类,ref变量与可null变量巧用,属性实际运用,唯一性验证,url传递中文编码和解码问题 讲解篇:1,服务端aspx,2,服务端后台返回数据(这里采用服务器端程序:aspx.cs)

mybatis generator自动生成 实体类, sqlmap配置文件 详细介绍

我使用的是Eclipse Luna 装了自己常用的插件, generator也是其中一个推荐下载 MyBatis_Generator_1.3.1.zip离线安装包 <?xml version="1.0" encoding="UTF-8"?> <!DOCTYPE generatorConfiguration PUBLIC "-//mybatis.org//DTD MyBatis Generator Configuration 1.0//EN

Hibernate自动生成实体类注解(转)

常用的hibernate annotation标签如下: @Entity --注释声明该类为持久类.将一个Javabean类声明为一 个实体的数据库表映射类,最好实现序列化.此时,默认情况下,所有的类属性都为映射到数据表的持久性字段.若在类中,添加另外属性,而非映射来数据库的, 要用下面的Transient来注解. @Table (name="promotion_info") --持久性映射的表(表名="promotion_info)[email protected]是类一级

Mybatis自动生成实体类和实体映射工具

Mybatis Mysql生成实体类 用到的Lib包: mybatis-generator-core-1.3.2.jarmysql-connector-java-5.1.30.jar 1. 创建一个文件generator.properties, 主要用于配置相关路径和数据库信息. #工程src路径 project = D:/project/ #工程存放mapper.xml路径 resource = D:/project/ #指定数据连接驱动jar地址 classPath=D:/project/m

SSM框架搭建(三) 数据库创建和MyBatis生成器自动生成实体类、DAO接口和Mapping映射文件

一:创建数据库 创建Student表 DROP TABLE IF EXISTS `Student`; CREATE TABLE `Student` ( `id` int(11) NOT NULL AUTO_INCREMENT, `user_name` varchar(40) NOT NULL, `password` varchar(255) NOT NULL, `age` int(4) NOT NULL, PRIMARY KEY (`id`) ) ENGINE=InnoDB AUTO_INCRE