那很有趣!
好吧,我真的希望那不是家庭作业。
事实证明,有一个非常简单的递归解决方案。在每一层你想要的是获取一个树列表,将它们成对地收集到更深的树中,然后在这一层附加新的叶子。这可以使用“foldr”来编写,但我认为它会不太清楚。
我应该澄清一下输入;在您提到的页面上,规格看起来像
在 0
级离开:在 1 级
离开:在 2 级离开:x23、x42、x23
在第 3 级离开:x24、x23
这将对应于输入
'(() () (x23 x42 x23) (x24 x23))
到下面的程序。
此外,这里唯一要做的就是将该表映射到二叉树,这只有在解码时才有帮助。对于编码,这个二叉树是没有用的。
最后,对如何设计程序大喊大叫;我仔细按照设计配方,点缀我所有的 i 并交叉我所有的 t。请先测试用例!
干杯!
约翰克莱门茨
#lang racket
(require rackunit)
;; a tree is either
;; a symbol, or
;; (list tree tree)
;; a specification is
;; (listof (listof symbol))
;; spec->tree : specification -> tree
;; run spec->treelist, ensure that it's a list of length 1, return it.
(define (spec->tree spec)
(match (spec->treelist spec)
[(list tree) tree]
[other (error 'spec->tree "multiple trees produced")]))
;; spec->treelist : specification -> (listof tree)
;; given a *legal* specification, produce
;; the corresponding tree. ONLY WORKS FOR LEGAL SPECIFICATIONS...
(define (spec->treelist spec)
(cond [(empty? spec) empty]
[else (append (first spec) (gather-pairs (spec->treelist (rest spec))))]))
;; go "up one level" by grouping each pair of trees into one tree.
;; The length of the list must be a number divisible by two.
(define (gather-pairs trees)
(match trees
[(list) empty]
[(list-rest a b remaining) (cons (list a b) (gather-pairs remaining))]
[other (error 'gather "improperly formed specification")]))
;; TEST CASES
(check-equal? (gather-pairs '(a b c d)) '((a b) (c d)))
(check-equal? (spec->treelist '((top))) '(top))
(check-equal? (spec->treelist '(() (two-a two-b))) '((two-a two-b)))
(check-equal? (spec->treelist '(() (two-a) (three-a three-b)))
'((two-a (three-a three-b))))
(check-equal? (spec->treelist '(() () (three-a three-b three-c) (four-a four-b)))
'(((three-a three-b) (three-c (four-a four-b)))))
(check-equal? (spec->tree '(() () (three-a three-b three-c) (four-a four-b)))
'((three-a three-b) (three-c (four-a four-b))))