随笔-341  评论-2670  文章-0  trackbacks-0
    以前为了开发KFP,特别学习了一下lambda calculus(也就是我的博客的标题啦)。lanbda calculus是一门神奇的语言,在计算机出现之前就已经被搞出来了。这门语言只有三种语法,然后可以用这个语法来构造整数(!!!)、布尔型和很多递归数据结构等。

    首先介绍一下语法。
    1、func arg代表一个函数调用,func是函数表达式,arg是参数。
    2、\param.value代表一个函数定义,参数是param,返回结果value。
    3、(expr)代表expr的优先级较高。

    上面就是所有的语法了。乍一看好像什么都没有,其实不然。我们现在先看一个东西:函数定义。
    为一个函数定义一个名称是很简单的:
    let double = \num.inc (inc num)) in ...
    代码“...”可以访问到函数double,但是函数定义的内部却不行。不过这并不会带来什么问题(其实是可以递归的,有办法)。当然let-in不是一个合法的lambda calculus程序,不过可以被翻译为:
    (\double. ...) (\num.inc (inc num))
    根据语法规则1,可以将后面的整个函数当作实参传入形参,将所有的double都替换成后面的那个东西,于是double的名称就被定下来了。

    递归怎么办呢?譬如说要写个函数sum n来计算1+2+...+n的值:
    let sum n = if (n==0) 0 (n+(sum (n-1))) in ...
    这里为了方便,我们假设所有的运算符都是存在的。其实a op b可以看成函数调用op a b,如果我们给每一个运算符都分配一个名字,实际上就可以用正确的lambda calculus语法来说明了。所以这里为了方便先这么干。

    sum内部是看不见sum的,因为翻译了之后变成:
    (\sum. ...) (if (n==0) 0 (n+(sum (n-1))))
    那怎么办呢?既然看不见自己,我可以让调用者再外部把它自己传进去当参数总可以吧:
    let sum n = \SELF. if (n==0) 0 (n + (SELF (n-1))) in ...
    但是我们不能用(sum sum)的方法来调用,因为到了最后(SELF (n-1))变成(sum (n-1)),下一步就错了。所以我们造了个Y函数:
    let Y = \f.(\t.f (t t)) (\t.f (t t)) in
    let sum = Y (\SELF. if (n==0) 0 (n + (SELF (n-1)))) in ...
    这样函数\SELF. if (n==0) 0 (n+(SELF (n-1))))就被Y变成了一个真正的递归函数了。不过在这里我不想花很大篇幅解释Y是怎么来的,有兴趣的读者看这里

    于是我们可以这么定义数字:
    zero = \s.\z.z
    one = \s.\z.s z
    two = \s.\z.s (s z)
    three = \s.\z.s (s (s z))
    four = \s.\z.s (s (s (s z)))
    数字n是一个函数,这个函数接受两个参数s和z,返回结果是拿s在z上应用n次的结果。所以我们可以很方便的实现乘法:拿“加法”函数当成第一个参数传进a,然后去应用b,就变成乘法了。我们首先创造一个加1函数:
    inc = \a.\s.\z.a s (s z)
    然后是加法和乘法:
    add = \a.\b.\s.\z.a s (b s z)
    mul = \a.\b.\s.\z.a (b s) z
    所以对代码mul (add (one two)) (add (three four))进行求值的时候,我们得到(1+2)*(3+4)=21:\s.\z.(s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s z)))))))))))))))))))))。

    布尔值也是一样的。我们让true接受两个参数返回第一个,false接受两个参数返回第二个:
    true = \a.\b.a
    false = \a.\b.b
    那么and or xor not可以分别写成:
    and = \a.\b.a b false
    or = \a.\b.a true b
    not = \a.a false true
    xor = \a.\b.a (not b) b
    函数if cond t f跟cond t f等价,所以我们不需要if函数。

    接下来怎么实现数据结构呢?假设我们实现一个pair,让pair 1 2将数字保存起来,让first(pair 1 2)返回1,second(pair 1 2)返回2:
    pair = \a.\b.\c.c a b
    first = \p.p true
    second = \p.p false
    我们举一个例子,让p=pair 1 (pair 2 3),然后求first (second p):
    first (second p) =
    first (second (pair 1 (pair 2 3))) =
    first ((pair 1(pair 2 3)) false) =
    first (false 1 (pair 2 3)) =
    first (pair 2 3) =
    (pair 2 3) true =
    true 2 3 =
    2
    是不是很神奇捏,lambda calculus仅需那么几条语法就可以实现所有东西了。将pair嵌套在一起可以构成一个列表。下面我们来写一个完整的程序,构造列表[1,2,3,4,5],然后对每一个元素求平方后相加:1*1 + 2*2 + 3*3 + 4*4 + 5*5 = 55:

    下面是完整的程序:
 1 let Y = \f.(\t.f (t t)) (\t.f (t t)) in
 2 
 3 let true = \a.\b.a in
 4 let false = \a.\b.b in
 5 let and = \a.\b.a b false in
 6 let or = \a.\b.a true b in
 7 let not = \a.a false true in
 8 let xor = \a.\b.a (not b) b in
 9 
10 let zero = \s.\z.z in
11 let inc = \a.\s.\z.a s (s z) in
12 let one = inc zero in
13 let two = inc one in
14 let three = inc two in
15 let four = inc three in
16 let five = inc four in
17 let six = inc five in
18 let seven = inc six in
19 let eight = inc seven in
20 let nine = inc eight in
21 let ten = inc nine in
22 let add = \a.\b.\s.\z.a s (b s z) in
23 let mul = \a.\b.\s.\z.a (b s) z in
24 let sqr = \a.mul a a in
25 
26 let pair = \a.\b.\c.c a b in
27 let first = \p.p true in
28 let second = \p.p false in
29 let empty = pair false false in
30 let list = \a.\b.pair true (pair a b) in
31 let head = \xs.(first xs) (first (second xs)) ERROR in
32 let tail = \xs.(first xs) (second (second xs)) ERROR in
33 let join = Y \SELF.\xs.\ys.(first xs) (list (head xs) (SELF (tail xs) ys)) ys in
34 let trans = Y \SELF.\f.\xs.(first xs) (list (f (head xs)) (SELF f (tail xs))) empty in
35 
36 let foldl = Y \SELF.\op.\init.\xs.(first xs) (SELF op (op init (head xs)) (tail xs)) init in
37 let foldr = Y \SELF.\op.\init.\xs.(first xs) (op (head xs) (SELF op init (tail xs))) init in
38 let length = foldl (\a.\b.inc a) zero in
39 let sum = foldl add zero in
40 
41 let long_list = list one (list two (list three (list four (list five empty)))) in
42 
43 sum (trans sqr long_list)

    然后是结果。首先我的lambda calculus虚拟机把let-in按照上面的方法重新转换为标准的lambda calculus程序,最后求值:


    数一数吧,上面有55个"(s",所以结果就是55了。

    我们添加一个计算a的b次方的函数:pow = \a.\b.b (mul a) one,将最后几行改成:
    let one_to_five = list one (list two (list three (list four (list five empty)))) in
    let one_to_ten = join one_to_five (trans (add five) one_to_five) in
    let long_list = trans (pow two) one_to_ten in
    sum long_list
    这计算2^1 + 2^2 + ... + 2^9 + 2^10。结果太长,不截屏幕了,直接复制出来:

  1 (\Y.(\true.(\false.(\and.(\or.(\not.(\xor.(\zero.(\one.(\two.(\three.(\four.(\fi
  2 ve.(\six.(\seven.(\eight.(\nine.(\ten.(\inc.(\add.(\mul.(\pow.(\sqr.(\pair.(\fir
  3 st.(\second.(\empty.(\list.(\head.(\tail.(\join.(\trans.(\foldl.(\foldr.(\length
  4 .(\sum.(\one_to_five.(\one_to_ten.(\long_list.(sum long_list) ((trans (pow two))
  5  one_to_ten)) ((join one_to_five) ((trans (add five)) one_to_five))) ((list one)
  6  ((list two) ((list three) ((list four) ((list five) empty)))))) ((foldl add) ze
  7 ro)) ((foldl \a.\b.(inc a)) zero)) (Y \SELF.\op.\init.\xs.(((first xs) ((op (hea
  8 d xs)) (((SELF op) init) (tail xs)))) init))) (Y \SELF.\op.\init.\xs.(((first xs
  9 ) (((SELF op) ((op init) (head xs))) (tail xs))) init))) (Y \SELF.\f.\xs.(((firs
 10 t xs) ((list (f (head xs))) ((SELF f) (tail xs)))) empty))) (Y \SELF.\xs.\ys.(((
 11 first xs) ((list (head xs)) ((SELF (tail xs)) ys))) ys))) \xs.(((first xs) (seco
 12 nd (second xs))) ERROR)) \xs.(((first xs) (first (second xs))) ERROR)) \a.\b.((p
 13 air true) ((pair a) b))) ((pair falsefalse)) \p.(p false)) \p.(p true)) \a.\b.
 14 \c.((c a) b)) \a.((mul a) a)) \a.\b.((b (mul a)) one)) \a.\b.((b (add a)) zero))
 15  \a.\b.((b inc) a)) \a.\s.\z.((a s) (s z))) \s.\z.(s (s (s (s (s (s (s (s (s (s
 16 z))))))))))) \s.\z.(s (s (s (s (s (s (s (s (s z)))))))))) \s.\z.(s (s (s (s (s (
 17 s (s (s z))))))))) \s.\z.(s (s (s (s (s (s (s z)))))))) \s.\z.(s (s (s (s (s (s
 18 z))))))) \s.\z.(s (s (s (s (s z)))))) \s.\z.(s (s (s (s z))))) \s.\z.(s (s (s z)
 19 ))) \s.\z.(s (s z))) \s.\z.(s z)) \s.\z.z) \a.\b.((a (not b)) b)) \a.((a false)
 20 true)) \a.\b.((a true) b)) \a.\b.((a b) false)) \a.\b.b) \a.\b.a) \f.(\t.(f (t t
 21 )) \t.(f (t t))))
 22 最终结果:
 23 \s.\z.(s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 24  (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (
 25 s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 26 (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 27  (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (
 28 s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 29 (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 30  (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (
 31 s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 32 (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 33  (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (
 34 s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 35 (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 36  (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (
 37 s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 38 (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 39  (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (
 40 s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 41 (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 42  (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (
 43 s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 44 (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 45  (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (
 46 s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 47 (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 48  (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (
 49 s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 50 (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 51  (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (
 52 s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 53 (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 54  (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (
 55 s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 56 (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 57  (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (
 58 s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 59 (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 60  (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (
 61 s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 62 (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 63  (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (
 64 s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 65 (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 66  (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (
 67 s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 68 (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 69  (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (
 70 s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 71 (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 72  (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (
 73 s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 74 (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 75  (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (
 76 s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 77 (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 78  (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (
 79 s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 80 (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 81  (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (
 82 s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 83 (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 84  (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (
 85 s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 86 (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 87  (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (
 88 s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 89 (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 90  (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (
 91 s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 92 (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 93  (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (
 94 s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 95 (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 96  (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (
 97 s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 98 (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s
 99  (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s (s z)))))))))))))))
100 ))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))
101 ))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))
102 ))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))
103 ))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))
104 ))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))
105 ))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))
106 ))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))
107 ))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))
108 ))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))
109 ))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))
110 ))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))
111 ))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))
112 ))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))
113 ))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))
114 ))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))
115 ))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))
116 ))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))
117 ))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))
118 ))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))
119 ))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))
120 ))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))
121 ))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))
122 ))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))
123 ))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))
124 ))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))
125 )))))))))))))))))))))))))))))))


    下面是lambda calculus虚拟机的C++代码。代码使用了Vczh Combinator Parser组装编译器,然后用一个池来存放运行时需要的东西,速度很快:

  1 #include "..\..\..\..\VL++\Library\Platform\VL_Console.h"
  2 #include "..\..\..\..\VL++\Library\Data\Data\VL_Data_Map.h"
  3 #include "..\..\..\..\VL++\Library\Data\Data\VL_Data_Pool.h"
  4 #include "..\..\..\..\VL++\Library\Data\VL_Stream.h"
  5 #include "..\..\..\..\VL++\Library\Data\VL_System.h"
  6 #include "..\..\..\..\VL++\Library\Data\Grammar2\Combinator\VL_CpKernel.h"
  7 #include "..\..\..\..\VL++\Library\Data\Grammar2\Combinator\VL_CpLexer.h"
  8 
  9 using namespace vl;
 10 using namespace vl::platform;
 11 using namespace vl::stream;
 12 using namespace vl::system;
 13 using namespace vl::pool;
 14 using namespace vl::grammar;
 15 
 16 /*********************************************************************************************************
 17 运行时表达式
 18 *********************************************************************************************************/
 19 
 20 enum LambdaRuntimeKind
 21 {
 22     lrkPrimitive,
 23     lrkClosure,
 24     lrkInvoke
 25 };
 26 class LambdaRuntime
 27 {
 28 public:
 29     typedef VL_Pool<LambdaRuntime>            RuntimePool;
 30 
 31     RuntimePool*                Pool;
 32     LambdaRuntimeKind            Kind;
 33     VInt                        ID;
 34     LambdaRuntime*                Closure;
 35     LambdaRuntime*                Expression;
 36 
 37     void SetPrimitive(RuntimePool* aPool , VInt aID)
 38     {
 39         Pool=aPool;
 40         Kind=lrkPrimitive;
 41         ID=aID;
 42         Closure=0;
 43         Expression=0;
 44     }
 45 
 46     void SetClosure(RuntimePool* aPool , VInt aID , LambdaRuntime* aExpression)
 47     {
 48         Pool=aPool;
 49         Kind=lrkClosure;
 50         ID=aID;
 51         Closure=0;
 52         Expression=aExpression;
 53     }
 54 
 55     void SetInvoke(RuntimePool* aPool , LambdaRuntime* aClosure , LambdaRuntime* aExpression)
 56     {
 57         Pool=aPool;
 58         Kind=lrkInvoke;
 59         ID=-1;
 60         Closure=aClosure;
 61         Expression=aExpression;
 62     }
 63 
 64     LambdaRuntime* Alpha(VInt ExpID , LambdaRuntime* Code)
 65     {
 66         switch(Kind)
 67         {
 68         case lrkPrimitive:
 69             return (ID==ExpID)?Code:this;
 70         case lrkClosure:
 71             if(ID==ExpID)
 72             {
 73                 return this;
 74             }
 75             else
 76             {
 77                 LambdaRuntime* NewExpression=Expression->Alpha(ExpID,Code);
 78                 if(Expression==NewExpression)
 79                 {
 80                     return this;
 81                 }
 82                 else
 83                 {
 84                     LambdaRuntime* AlphaTransformed=Pool->Alloc();
 85                     AlphaTransformed->SetClosure(Pool,ID,NewExpression);
 86                     return AlphaTransformed;
 87                 }
 88             }
 89         case lrkInvoke:
 90             {
 91                 LambdaRuntime* NewClosure=Closure->Alpha(ExpID,Code);
 92                 LambdaRuntime* NewExpression=Expression->Alpha(ExpID,Code);
 93                 if(Closure==NewClosure && Expression==NewExpression)
 94                 {
 95                     return this;
 96                 }
 97                 else
 98                 {
 99                     LambdaRuntime* AlphaTransformed=Pool->Alloc();
100                     AlphaTransformed->SetInvoke(Pool,NewClosure,NewExpression);
101                     return AlphaTransformed;
102                 }
103             }
104         default:
105             return 0;
106         }
107     }
108 
109     LambdaRuntime* Evaluate(VBool EvaluatingRoot)
110     {
111         switch(Kind)
112         {
113         case lrkPrimitive:
114             return this;
115         case lrkClosure:
116             if(EvaluatingRoot)
117             {
118                 Expression=Expression->Evaluate(true);
119                 return this;
120             }
121             else
122             {
123                 return this;
124             }
125         case lrkInvoke:
126             {
127                 Closure=Closure->Evaluate(false);
128                 if(Closure->Kind==lrkClosure)
129                 {
130                     *this=*Closure->Expression->Alpha(Closure->ID,Expression)->Evaluate(EvaluatingRoot);
131                     return this;
132                 }
133                 else
134                 {
135                     Expression=Expression->Evaluate(EvaluatingRoot);
136                     return this;
137                 }
138             }
139         default:
140             return 0;
141         }
142     }
143 };
144 
145 class LambdaEnvironment
146 {
147 public:
148     LambdaRuntime::RuntimePool    Pool;
149 
150     LambdaEnvironment():Pool(65536)
151     {
152     }
153 };
154 
155 /*********************************************************************************************************
156 语法树
157 *********************************************************************************************************/
158 
159 class LambdaError
160 {
161 public:
162     VUnicodeString Message;
163 
164     LambdaError(VUnicodeString aMessage)
165     {
166         Message=aMessage;
167     }
168 };
169 
170 class LambdaIdentifier
171 {
172 public:
173     typedef VL_ListedMap<VUnicodeString , VInt>        TokenMap;
174     typedef VL_ListedMap<VInt , VUnicodeString>        TokenMapRev;
175 
176     VUnicodeString            Token;
177     VInt                    ID;
178 
179     LambdaIdentifier()
180     {
181         ID=-1;
182     }
183 
184     VBool operator==(const LambdaIdentifier& Identifier)
185     {
186         return ID==Identifier.ID;
187     }
188 
189     VBool operator!=(const LambdaIdentifier& Identifier)
190     {
191         return ID!=Identifier.ID;
192     }
193 
194     void Initialize(TokenMap& Tokens)
195     {
196         VInt Index=Tokens.IndexOfKey(Token);
197         if(Index==-1)
198         {
199             ID=Tokens.KeyCount();
200             Tokens.Add(Token,ID);
201         }
202         else
203         {
204             ID=Tokens.ValueOfIndex(Index);
205         }
206     }
207 
208     void Initialize(TokenMapRev& Tokens)
209     {
210         Token=Tokens[ID];
211     }
212 
213     void Collect(TokenMapRev& Tokens)
214     {
215         Tokens.Add(ID,Token);
216     }
217 };
218 
219 class LambdaCode : public VL_Base
220 {
221 public:
222     typedef VL_AutoPtr<LambdaCode>                            Ptr;
223 
224     friend LambdaCode::Ptr        CreateCode(LambdaRuntime* Runtime);
225 
226     virtual void                Initialize(LambdaIdentifier::TokenMap& Tokens)=0;
227     virtual void                Initialize(LambdaIdentifier::TokenMapRev& Tokens)=0;
228     virtual void                Collect(LambdaIdentifier::TokenMapRev& Tokens)=0;
229     virtual VUnicodeString        ToString()=0;
230     virtual LambdaRuntime*        CreateRuntime(LambdaEnvironment* Environment)=0;
231 
232     LambdaCode::Ptr Evaluate()
233     {
234         LambdaEnvironment Environment;
235         LambdaRuntime* Runtime=CreateRuntime(&Environment);
236         LambdaRuntime* Evaluated=Runtime->Evaluate(true);
237         LambdaCode::Ptr Code=CreateCode(Evaluated);
238         LambdaIdentifier::TokenMapRev Tokens;
239         Collect(Tokens);
240         Code->Initialize(Tokens);
241         return Code;
242     }
243 };
244 
245 class LambdaCodePrimitive : public LambdaCode
246 {
247 public:
248     LambdaIdentifier        Name;
249 
250     void Initialize(LambdaIdentifier::TokenMap& Tokens)
251     {
252         Name.Initialize(Tokens);
253     }
254 
255     void Initialize(LambdaIdentifier::TokenMapRev& Tokens)
256     {
257         Name.Initialize(Tokens);
258     }
259 
260     void Collect(LambdaIdentifier::TokenMapRev& Tokens)
261     {
262         Name.Collect(Tokens);
263     }
264 
265     VUnicodeString ToString()
266     {
267         return Name.Token;
268     }
269 
270     LambdaRuntime* CreateRuntime(LambdaEnvironment* Environment)
271     {
272         LambdaRuntime* Runtime=Environment->Pool.Alloc();
273         Runtime->SetPrimitive(&Environment->Pool,Name.ID);
274         return Runtime;
275     }
276 };
277 
278 class LambdaCodeClosure : public LambdaCode
279 {
280 public:
281     LambdaIdentifier        Parameter;
282     LambdaCode::Ptr            Expression;
283 
284     void Initialize(LambdaIdentifier::TokenMap& Tokens)
285     {
286         Parameter.Initialize(Tokens);
287         Expression->Initialize(Tokens);
288     }
289 
290     void Initialize(LambdaIdentifier::TokenMapRev& Tokens)
291     {
292         Parameter.Initialize(Tokens);
293         Expression->Initialize(Tokens);
294     }
295 
296     void Collect(LambdaIdentifier::TokenMapRev& Tokens)
297     {
298         Parameter.Collect(Tokens);
299         Expression->Collect(Tokens);
300     }
301 
302     VUnicodeString ToString()
303     {
304         return L"\\"+Parameter.Token+L"."+Expression->ToString();
305     }
306 
307     LambdaRuntime* CreateRuntime(LambdaEnvironment* Environment)
308     {
309         LambdaRuntime* Runtime=Environment->Pool.Alloc();
310         Runtime->SetClosure(&Environment->Pool,Parameter.ID,Expression->CreateRuntime(Environment));
311         return Runtime;
312     }
313 };
314 
315 class LambdaCodeInvoke : public LambdaCode
316 {
317 public:
318     LambdaCode::Ptr            Function;
319     LambdaCode::Ptr            Argument;
320 
321     void Initialize(LambdaIdentifier::TokenMap& Tokens)
322     {
323         Function->Initialize(Tokens);
324         Argument->Initialize(Tokens);
325     }
326 
327     void Initialize(LambdaIdentifier::TokenMapRev& Tokens)
328     {
329         Function->Initialize(Tokens);
330         Argument->Initialize(Tokens);
331     }
332 
333     void Collect(LambdaIdentifier::TokenMapRev& Tokens)
334     {
335         Function->Collect(Tokens);
336         Argument->Collect(Tokens);
337     }
338 
339     VUnicodeString ToString()
340     {
341         return L"("+Function->ToString()+L" "+Argument->ToString()+L")";
342     }
343 
344     LambdaRuntime* CreateRuntime(LambdaEnvironment* Environment)
345     {
346         LambdaRuntime* Runtime=Environment->Pool.Alloc();
347         Runtime->SetInvoke(&Environment->Pool,Function->CreateRuntime(Environment),Argument->CreateRuntime(Environment));
348         return Runtime;
349     }
350 };
351 
352 LambdaCode::Ptr CreateCode(LambdaRuntime* Runtime)
353 {
354     switch(Runtime->Kind)
355     {
356     case lrkPrimitive:
357         {
358             LambdaCodePrimitive* Primitive=new LambdaCodePrimitive;
359             Primitive->Name.ID=Runtime->ID;
360             return Primitive;
361         }
362     case lrkClosure:
363         {
364             LambdaCodeClosure* Closure=new LambdaCodeClosure;
365             Closure->Parameter.ID=Runtime->ID;
366             Closure->Expression=CreateCode(Runtime->Expression);
367             return Closure;
368         }
369     case lrkInvoke:
370         {
371             LambdaCodeInvoke* Invoke=new LambdaCodeInvoke;
372             Invoke->Function=CreateCode(Runtime->Closure);
373             Invoke->Argument=CreateCode(Runtime->Expression);
374             return Invoke;
375         }
376     default:
377         return 0;
378     }
379 }
380 
381 /*********************************************************************************************************
382 语法分析器
383 *********************************************************************************************************/
384 
385 enum LambdaTokenID
386 {
387     ltiLet,
388     ltiIn,
389     ltiClosure,
390     ltiName,
391     ltiLeft,
392     ltiRight,
393     ltiDot,
394     ltiEqual,
395 };
396 
397 LambdaCode::Ptr ToPrimitive(VL_CpToken Token)
398 {
399     LambdaCodePrimitive* Primitive=new LambdaCodePrimitive;
400     Primitive->Name.Token=VUnicodeString(Token.Start,Token.Length);
401     return Primitive;
402 }
403 
404 LambdaCode::Ptr ToInvoke(const VL_CpList<LambdaCode::Ptr>& Input)
405 {
406     LambdaCode::Ptr Code=Input.Head->Data;
407     VL_CpList<LambdaCode::Ptr>::Node::Ptr Current=Input.Head->Next;
408     while(Current)
409     {
410         LambdaCodeInvoke* Invoke=new LambdaCodeInvoke;
411         Invoke->Function=Code;
412         Invoke->Argument=Current->Data;
413         Code=Invoke;
414         Current=Current->Next;
415     }
416     return Code;
417 }
418 
419 LambdaCode::Ptr ToClosure(const VL_CpPair<VL_CpToken , LambdaCode::Ptr>& Input)
420 {
421     LambdaCodeClosure* Closure=new LambdaCodeClosure;
422     Closure->Parameter.Token=VUnicodeString(Input.First.Start,Input.First.Length);
423     Closure->Expression=Input.Second;
424     return Closure;
425 }
426 
427 LambdaCode::Ptr ToLetIn(const VL_CpPair<VL_CpPair<VL_CpToken , LambdaCode::Ptr> , LambdaCode::Ptr>& Input)
428 {
429     LambdaCodeInvoke* Invoke=new LambdaCodeInvoke;
430     Invoke->Function=ToClosure(VL_CpPair<VL_CpToken , LambdaCode::Ptr>(Input.First.First,Input.Second));
431     Invoke->Argument=Input.First.Second;
432     return Invoke;
433 }
434 
435 LambdaCode::Ptr Parse(VUnicodeString Code)
436 {
437     VL_CpLexer Lexer;
438     Lexer
439         <<Token(false,L"let",ltiLet)
440         <<Token(false,L"in",ltiIn)
441         <<Token(false,L"\\",ltiClosure)
442         <<Token(false,L"(",ltiLeft)
443         <<Token(false,L")",ltiRight)
444         <<Token(false,L".",ltiDot)
445         <<Token(false,L"=",ltiEqual)
446         <<Token(false,_Name,ltiName)
447         <<Token(true,_Blank,-1)
448         <<Token(true,_CComment,-1)
449         <<Token(true,_CppComment,-1)
450         ;
451 
452     typedef _Wrapper<VL_CpTokenNodePtr , LambdaCode::Ptr> CodeRule;
453     typedef _Terminal<VL_CpTokenNodePtr , LambdaCode::Ptr> CodeTerminal;
454 
455     CodeRule Unit,Expr;
456 
457     CodeTerminal Primitive =
458         (ToPrimitive <<= Token(ltiName));
459 
460     CodeTerminal Closure = 
461         (ToClosure <<= ((Toks(L"\\"> Token(ltiName) < Toks(L".")) + Expr));
462 
463     CodeTerminal Bracket = 
464         (Toks(L"("> Expr < Toks(L")"));
465 
466     CodeTerminal InExpr =
467         Toks(L"in"> Expr;
468 
469     Unit=
470         Primitive || Closure || Bracket;
471 
472     CodeTerminal Invoke=
473         (ToInvoke <<= ++Unit);
474 
475     CodeTerminal LetIn=
476         (ToLetIn <<=((Toks(L"let"> Token(ltiName) < Toks(L"=")) + Expr + InExpr));
477 
478     Expr=
479         Invoke || LetIn;
480 
481     VL_CpParser<VL_CpTokenNodePtr , LambdaCode::Ptr> Parser=Expr;
482 
483     LambdaCode::Ptr Lambda=Parser.Parse(Lexer.Parse(Code.Buffer()).First.Head).Head->Data.First;
484 
485     LambdaIdentifier::TokenMap Tokens;
486     Lambda->Initialize(Tokens);
487     return Lambda;
488 }
489 
490 /*********************************************************************************************************
491 主程序
492 *********************************************************************************************************/
493 
494 void vlmain()
495 {
496     GetConsole()->SetTitle(L"Vczh Lambda Evaluator");
497     GetConsole()->SetTestMemoryLeaks(true);
498     GetConsole()->SetPauseOnExit(true);
499 
500     VUnicodeString WorkSpace=VFileName(GetConsole()->GetAppPath()).MakeAbsolute(L"..\\TestData\\").GetStrW();
501     VUnicodeString Code;
502     {
503         VL_FileStream Stream(WorkSpace+L"Program_01.txt",VL_FileStream::vomRead);
504         Code=ReadText(&Stream);
505     }
506 
507     try
508     {
509         LambdaCode::Ptr Lambda=Parse(Code);
510         GetConsole()->WriteLine(Lambda->ToString());
511         try
512         {
513             LambdaCode::Ptr Evaluated=Lambda->Evaluate();
514             GetConsole()->WriteLine(L"最终结果:");
515             GetConsole()->WriteLine(Evaluated->ToString());
516         }
517         catch(const LambdaError& Error)
518         {
519             GetConsole()->WriteLine(Error.Message);
520         }
521     }
522     catch(const VL_CpException<VL_CpTokenNodePtr>& e)
523     {
524         if(e.Input)
525         {
526             GetConsole()->WriteLine(L""+VUnicodeString(e.Input->Data.Line+1)+L"行,记号\""+VUnicodeString(e.Input->Data.Start,e.Input->Data.Length)+L"\"附近有语法错误。");
527         }
528         else
529         {
530             GetConsole()->WriteLine(L"程序末尾附近出现语法错误。");
531         }
532     }
533 }

posted on 2009-05-11 04:30 陈梓瀚(vczh) 阅读(5394) 评论(7)  编辑 收藏 引用 所属分类: 启示

评论:
# re: 丘奇数(Church Numerals)和lambda calculus 2009-05-11 05:54 | cy
500几行就诞生了一门语言.......  回复  更多评论
  
# re: 丘奇数(Church Numerals)和lambda calculus 2009-05-11 06:32 | suxiaojack
牛!
这个语法太酷了。
to cy :加上头文件不只500多行。
  回复  更多评论
  
# re: 丘奇数(Church Numerals)和lambda calculus 2009-05-11 08:21 | cy
那些头文件属于这个语言的“设计支持”部分,暂时不算进去~~

感觉这样的语言到达了电路设计级别  回复  更多评论
  
# re: 丘奇数(Church Numerals)和lambda calculus 2009-05-12 06:28 | yindf
127 行的Evaluate 参数是false,
236 行的Evaluate 参数是true,

可能会造成不一致吧?

236行的true表示应用序求值(先求值,后代换),但是127行的false表示不对前面的闭包求值,所以这个闭包是正则序求值的(先代换,后求值)。于是就可能造成不一致。

不知道我理解的对不对,请指教。  回复  更多评论
  
# re: 丘奇数(Church Numerals)和lambda calculus 2009-05-13 00:53 | 陈梓瀚(vczh)
@yindf
不是这个意思,Evaluate=false指的是尽可能少求值,譬如对于一个函数表达式来说,(\x.(\y.y) x) 1和(\x.x) 1在对被调用函数的处理上是没区别的,但是对于程序的结果来说,(\x.(\y.y) x)和(\x.x)是有区别的。所以Evaluate的意思其实是【正在求值的表达式是否根表达式】。如果\x.y是根表达式,那么y也是。如果\x.y z是跟表达式,那么y和z不算根表达式。  回复  更多评论
  
# re: 丘奇数(Church Numerals)和lambda calculus 2009-05-13 01:29 | yindf
@陈梓瀚(vczh)

其实问题是一样的,主要是你对表达式怎么求值的问题,比如:

\x.x 1+2 这个式子,你是要
先算1+2=3呢, (1)
还是把\x.x 1+2直接替换成1+2? (2)

如果你想做(1)的话,那么在(\x.(\y.y) x) 1中,(\x.(\y.y) x)就应该先被计算成 (\y.y),然后再\y.y 1

如果(2)的话,那么在(\x.(\y.y) x) 1中,先把x换成1,变成(\y.y) 1

(1)表示对所有函数,先求出参数表达式,然后再带入。
(2)表示先把参数直接带入展开,展开到最底层,最后一起求值。

在函数式编程里面,比如一个函数接受另一个函数做参数,那么你是要先计算参数的值呢,还是直接把参数带进去?

因为我感觉你在用(1)方法,所以我觉得函数在使用前应该求值。

其实问题的本质是函数在使用前应不应该被化简,对于采用应用序的语言来说,应该。  回复  更多评论
  
# re: 丘奇数(Church Numerals)和lambda calculus 2009-05-13 07:00 | 陈梓瀚(vczh)
@yindf
函数使用前不简化(只需要变成一个lambda表达式即可,lambda里面的内容不理),不然你无法对那个传说中的Y函数求值,会死循环的。你可以去学习一下惰性求值。  回复  更多评论
  

只有注册用户登录后才能发表评论。
网站导航: 博客园   IT新闻   BlogJava   博问   Chat2DB   管理