2019-03-13-stone测试

源自自制编译器+自制脚本语言+自制编程语言 三书比较?

两年多前只看了一本的前六章: 实践”两周自制脚本语言”一书【java吧】_百度贴吧

当时是希望一路添加测试用例, 并且最终把语言改为中文语法. 但看到Gluonj之后,似乎必须要Loader.run来执行解释器, 得不到返回值, 只能打印解释结果.

今天从此书官网(スクリプト言語の作り方)找到了它的附带源码库: chibash/stone, 顺便研究了一下Gluonj源码, 找到了它的JUnit辅助功能: https://github.com/chibash/gluonj/blob/master/src/javassist/gluonj/util/UTester.java

据此写了第五到十四章的测试, 运行效果见开头. 下面以第九章的为例:

测试用”石头”语言源码: 位置类

class Position {
    x = y = 0
    def move (nx, ny) {
        x = nx; y = ny;
    }
}
p = Position.new
p.move(3, 4)
p.x = 10
sum = p.x + p.y

测试代码: 面向对象解释器Test

package chap9;

import static org.junit.Assert.assertEquals;

import java.io.IOException;
import java.nio.charset.StandardCharsets;

import org.junit.Test;

import chap7.NestedEnv;
import chap8.Natives;
import javassist.gluonj.util.UTester;
import stone.ClassParser;
import stone.ParseException;
import stone.util.文件功用;
import stone.util.解释器功用;

public class 面向对象解释器Test extends ClassInterpreter {

  private static String 位置类 = "";
  static {
    try {
      位置类 = 文件功用.读文件("测试源码/chap9/位置类.txt", StandardCharsets.UTF_8);
    } catch (IOException e) {
    }
  }

  public static Object 求值(String 源代码) throws ParseException {
    return 解释器功用.求值(new ClassParser(), new Natives().environment(new NestedEnv()), 源代码);
  }

  @Test
  public void 例程() throws Throwable {
    if (UTester.runTestWith("chap9.ClassEvaluator", "chap8.NativeEvaluator", "chap7.ClosureEvaluator"))
      return;
    assertEquals(14, 求值(位置类));
  }
}

改写自BasicInterpreter解释器功用.求值()

  public static Object 求值(BasicParser 基本分析器, Environment 环境, String 源代码) throws ParseException {
    Lexer 词法分析器 = new Lexer(new StringReader(源代码));
    Object 终值 = null;
    while (词法分析器.peek(0) != Token.EOF) {
      ASTree  = 基本分析器.parse(词法分析器);
      if (!( instanceof NullStmnt)) {
        终值 = ((BasicEvaluator.ASTreeEx) ).eval(环境);
      }
    }
    return 终值;
}

又有了深入学习此书(顺便将代码中文化, 像之前的中文编程:手工翻译Vue.js源码:尝试重命名标识符与文本), 并据此开发一些中文编程语言原型的动力.

: 十四章中fib(33)的运行时间确实与Java的对应代码运行时间相差无几, 虽然如作者所言, 并不能以此判断语言的性能.