很高兴今天能发布这个声明:
经过不懈努力,Lysee于今晨8:30释出1.1.0.5511版本,新版本初步实现了语法定义,帮助Lysee又向前“
挪了”一步!
首先向Scheme/Lisp、Haskell、Perl、Python还有其它语言表示一下尊敬:
syntax void thanks(%(you) AND %(he))
{
= "thanks!", you, "and", he;
}
thanks "scheme/lisp" AND "haskell, python, perl and others";
“
syntax”是为Lysee新定义的关键字,用于向语法分析器提示有必要在后续的代码中检测并使用新定义的语法。
用“
syntax”定义的语法只是一个再普通不过的函数,下面是前面定义的“thanks”语法的伪代码:
public variant main(variant ARGS)
{
PUSH_FUNC main::thanks //==> 将函数thanks压栈
PUSH_STR scheme/lisp
PUSH_STR "haskell, python, perl and others"
CALL [3]
POP
RETURN
}
syntax void thanks(%(you) AND %(he)) //==> public void thanks(variant you he)
{
PUSH_STR thanks!
PUSH_VARB you
PUSH_STR and
PUSH_VARB he
PRINT [4]
}
原理:
编译器将函数定义中
%(ID)和
%{ID}格式中的标识符登记为
variant类型的参数,其它符号作为占位符被抛掉。
%(ID) - 表示直接计算
%(ID)所在位置的表达式,仅将结果传递给新函数。
%{ID} - 表示在编译时将
%{ID}对应位置上的表达式转换为闭包。
下面定义一个略微复杂的语法,实现
Pascal中的
repeat ... until循环:
// 定义 repeat .. until 语法
syntax void repeat(%(STATEMENT) until %{FALSE})
{
do { STATEMENT() } while (not FALSE());
}
// 使用
int guess, magic = 5;
repeat {
guess = sys::random(10);
= guess, eol;
} until (guess == magic);
// 输出
7
0
3
4
2
5
看到这儿,研究FP的朋友们应该可以会心的笑了,看看下面的伪代码就更清楚了
public variant main(variant ARGS)
{
VARB guess: int
VARB magic: int
PUSH_INT 5
SAVE_TO magic
POP
PUSH_FUNC main::repeat //==> repeat ... until
PUSH_SUBF main::main.1 //==> STATEMENT
PUSH_SUBF main::main.2 //==> FALSE
CALL [3]
POP
RETURN
}
public variant main.1() //==> STATEMENT
{
PUSH_FUNC sys::random
PUSH_INT 10
CALL [2]
SAVE_TO guess
POP
PUSH_VARB guess
PUSH EOL
PRINT [2]
RETURN
}
public variant main.2() //==> FALSE
{
PUSH_VARB guess
PUSH_VARB magic
CALC ==
RETURN [1]
}
syntax void repeat(%(STATEMENT) until %{FALSE})
{
0000:PUSH_VARB STATEMENT //==> main.1
CALL [1]
POP
PUSH_VARB FALSE //==> main.2
CALL [1]
CALC NOT
JMPT 0000: POP
}
再举一个常见的
for_each语法定义,真是很简单:
syntax void for_each(%(LIST): %(PROC)) // 要求LIST必须支持each操作
{
LIST.each(PROC);
}
// 调用
for_each strlist("hello\nworld\n!"): {|string item| = item, eol};
// 输出
hello
world
!
语法定义扎根在函数闭包上,怎么优化函数闭包的生成、使用和释放,如何节省系统资源仍然还是个大问题。
总结一下:语法定义是个好东西,但使用的代价可能不菲,特别是过分滥用时会破坏我们已有的代码。