|
@@ -0,0 +1,438 @@
|
|
|
+* 2.2 Hierarchical Data and the Closure Property
|
|
|
+
|
|
|
+!! "it is better to have 100 functions operate on one data structure
|
|
|
+ than 10 functions operate on 10 data structures" -- alan perlis
|
|
|
+
|
|
|
+Box and pointer diagram of (cons 1 2)
|
|
|
+
|
|
|
+: | +-----+-----+
|
|
|
+: +--->| | | +---+
|
|
|
+: | * | *------>| 2 |
|
|
|
+: +--|--+-----+ +---+
|
|
|
+: |
|
|
|
+: v
|
|
|
+: +---+
|
|
|
+: | 1 |
|
|
|
+: +---+
|
|
|
+
|
|
|
+pairs provide a universal building block from which we can build all
|
|
|
+sorts of data structures
|
|
|
+
|
|
|
+ - closure property :: the ability for pairing pairs
|
|
|
+ - hierarchical structures :: structures made up of parts that are
|
|
|
+ made up of parts that are made up of parts...
|
|
|
+
|
|
|
+note: closure in this sense is not the same as the closure used to
|
|
|
+describe procedures with free variables. the authors do not like the
|
|
|
+use of the term to mean this other thing.
|
|
|
+
|
|
|
+: the sequence 1, 2, 3, 4 repr as a chain of pairs
|
|
|
+
|
|
|
+: +---+---+ +---+---+ +---+---+ +---+---+
|
|
|
+: --->| * | *---->| * | *---->| * | *---->| * | X |
|
|
|
+: +-|-+---+ +-|-+---+ +-|-+---+ +-|-+---+
|
|
|
+: v v v v
|
|
|
+: +---+ +---+ +---+ +---+
|
|
|
+: | 1 | | 2 | | 3 | | 4 |
|
|
|
+: +---+ +---+ +---+ +---+
|
|
|
+
|
|
|
+** 2.2.1 Representing Sequences
|
|
|
+
|
|
|
+one useful structure is the *sequence*, ie a list
|
|
|
+
|
|
|
+#+BEGIN_SRC scheme
|
|
|
+ (cons 1
|
|
|
+ (cons 2
|
|
|
+ (cons 3
|
|
|
+ (cons 4 nil))))
|
|
|
+#+END_SRC
|
|
|
+
|
|
|
+scheme provides a primitive for this action
|
|
|
+
|
|
|
+#+BEGIN_SRC scheme
|
|
|
+ (list 1 2 3 4)
|
|
|
+ (define one-thru-four (list 1 2 3 4))
|
|
|
+ (car one-thru-four)
|
|
|
+ (cdr one-thru-four)
|
|
|
+#+END_SRC
|
|
|
+
|
|
|
+#+RESULTS:
|
|
|
+| 2 | 3 | 4 |
|
|
|
+
|
|
|
+#+BEGIN_SRC scheme
|
|
|
+ (define one-thru-four (list 1 2 3 4))
|
|
|
+ (cons 10 one-thru-four)
|
|
|
+ (cadr one-thru-four)
|
|
|
+#+END_SRC
|
|
|
+
|
|
|
+#+RESULTS:
|
|
|
+: 2
|
|
|
+
|
|
|
+list processing (lisping!) is often achieved by "cdring down" lists
|
|
|
+
|
|
|
+#+BEGIN_SRC scheme
|
|
|
+ (define (list-ref items n)
|
|
|
+ (if (= n 0)
|
|
|
+ (car items)
|
|
|
+ (list-ref (cdr items) (- n 1))))
|
|
|
+ (define squares (list 1 4 9 16 25))
|
|
|
+ (list-ref squares 4)
|
|
|
+#+END_SRC
|
|
|
+
|
|
|
+#+RESULTS:
|
|
|
+: 25
|
|
|
+
|
|
|
+we often cdr down the entire list. scheme offeres the primitve ~null?~
|
|
|
+to help
|
|
|
+#+BEGIN_SRC scheme
|
|
|
+ (define (length items)
|
|
|
+ (if (null? items)
|
|
|
+ 0
|
|
|
+ (+ 1 (length (cdr items)))))
|
|
|
+ (define odds (list 1 3 5 7 9))
|
|
|
+ (length odds)
|
|
|
+#+END_SRC
|
|
|
+
|
|
|
+if iteration is more to taste,
|
|
|
+#+BEGIN_SRC scheme
|
|
|
+ (define (length items)
|
|
|
+ (define (length-iter a count)
|
|
|
+ (if (null? a)
|
|
|
+ count
|
|
|
+ (length-iter (cdr a) (+ 1 count))))
|
|
|
+ (length-iter items 0))
|
|
|
+ (length (list 8 6 7 5 3 0 9))
|
|
|
+#+END_SRC
|
|
|
+
|
|
|
+another common technique is to "cons up" a list of answers while
|
|
|
+"cdring down" a list
|
|
|
+#+BEGIN_SRC scheme
|
|
|
+ (define (myappend list1 list2)
|
|
|
+ (if (null? list1)
|
|
|
+ list2
|
|
|
+ (cons (car list1) (myappend (cdr list1) list2))))
|
|
|
+ (myappend (list 1 2 3 4) (list 0 9 8 7))
|
|
|
+#+END_SRC
|
|
|
+
|
|
|
+#+RESULTS:
|
|
|
+| 1 | 2 | 3 | 4 | 0 | 9 | 8 | 7 |
|
|
|
+
|
|
|
+
|
|
|
+-- MAPPING OVER LISTS --
|
|
|
+
|
|
|
+a very useful operation is applying a transformation to each element
|
|
|
+in a list
|
|
|
+#+BEGIN_SRC scheme
|
|
|
+ (define (scale-list items factor)
|
|
|
+ (if (null? items)
|
|
|
+ (list)
|
|
|
+ (cons (* (car items) factor)
|
|
|
+ (scale-list (cdr items) factor))))
|
|
|
+ (scale-list (list 1 2 3 4 5) 100)
|
|
|
+#+END_SRC
|
|
|
+
|
|
|
+#+RESULTS:
|
|
|
+| 100 | 200 | 300 | 400 | 500 |
|
|
|
+
|
|
|
+we can abstract this pattern
|
|
|
+#+BEGIN_SRC scheme
|
|
|
+ (define (mymap proc items)
|
|
|
+ (if (null? items)
|
|
|
+ (list)
|
|
|
+ (cons (proc (car items))
|
|
|
+ (mymap proc (cdr items)))))
|
|
|
+ (mymap abs (list -10 2.5 -11.6 17))
|
|
|
+#+END_SRC
|
|
|
+
|
|
|
+#+RESULTS:
|
|
|
+| 10 | 2.5 | 11.6 | 17 |
|
|
|
+
|
|
|
+scheme provides a map primitive
|
|
|
+
|
|
|
+we can use map to redefine scale-list
|
|
|
+#+BEGIN_SRC scheme
|
|
|
+ (define (scale-list items factor)
|
|
|
+ (map (lambda (x) (* x factor))
|
|
|
+ items))
|
|
|
+#+END_SRC
|
|
|
+
|
|
|
+the original definition of scale-list drew attention to the
|
|
|
+item-by-item processing. map establishes a higher level of
|
|
|
+abstraction.
|
|
|
+
|
|
|
+!! map abstracts the impl of procedures that operate on lists from the
|
|
|
+ details of how elements are extracted and combined
|
|
|
+
|
|
|
+we are going to look at how this use of sequences provides a framework
|
|
|
+for organizing programs ~~ cool
|
|
|
+
|
|
|
+** 2.2.2 Hierarchical Structures
|
|
|
+representing sequences as lists generalizes to representing sequences
|
|
|
+made of sequences
|
|
|
+
|
|
|
+for example,
|
|
|
+: ((1 2) 3 4) ;; made by
|
|
|
+: (cons (list 1 2) (list 3 4))
|
|
|
+is a list of three items, the first of which is a list
|
|
|
+
|
|
|
+: ((1 2) 3 4) --> |.|.|---------->|.|.|-->|.|\|
|
|
|
+: | | |
|
|
|
+: v v v
|
|
|
+: |.|.|-->|.|\| 3 4
|
|
|
+: | |
|
|
|
+: v v
|
|
|
+: 1 2
|
|
|
+
|
|
|
+!! sequences of sequences are really /trees/
|
|
|
+
|
|
|
+to count the leaves in a tree:
|
|
|
+#+BEGIN_SRC scheme
|
|
|
+ (define x (cons (list 1 2) (list 3 4)))
|
|
|
+ (length x) ;; 3
|
|
|
+ (count-leaves x) ;; 4
|
|
|
+ (list x x) ;; (((1 2) 3 4) ((1 2) 3 4))
|
|
|
+ (length (list x x)) ;; 2
|
|
|
+ (count-leaves (list x x)) ;; 8
|
|
|
+
|
|
|
+ (define (count-leaves x)
|
|
|
+ (cond ((null? x) 0)
|
|
|
+ ((not (pair? x)) 1)
|
|
|
+ (else (+ (count-leaves (car x))
|
|
|
+ (count-leaves (cdr x))))))
|
|
|
+ (count-leaves x)
|
|
|
+#+END_SRC
|
|
|
+
|
|
|
+MAPPING OVER TREES
|
|
|
+
|
|
|
+#+BEGIN_SRC scheme
|
|
|
+ (define (scale-tree tree factor)
|
|
|
+ (cond ((null? tree) (list))
|
|
|
+ ((not (pair? tree)) (* tree factor))
|
|
|
+ (else (cons (scale-tree (car tree) factor)
|
|
|
+ (scale-tree (cdr tree) factor)))))
|
|
|
+ (scale-tree (list 1 (list 2 (list 3 4) 5) (list 6 7)) 10)
|
|
|
+#+END_SRC
|
|
|
+
|
|
|
+#+RESULTS:
|
|
|
+| 10 | (20 (30 40) 50) | (60 70) |
|
|
|
+
|
|
|
+#+BEGIN_SRC scheme
|
|
|
+ (define (scale-tree tree factor)
|
|
|
+ (map (lambda (sub-tree)
|
|
|
+ (if (pair? sub-tree)
|
|
|
+ (scale-tree sub-tree factor)
|
|
|
+ (* sub-tree factor)))
|
|
|
+ tree))
|
|
|
+ (scale-tree (list 1 (list 2 (list 3 4) 5) (list 6 7)) 2)
|
|
|
+#+END_SRC
|
|
|
+
|
|
|
+#+RESULTS:
|
|
|
+| 2 | (4 (6 8) 10) | (12 14) |
|
|
|
+
|
|
|
+** 2.2.3 Sequences as Conventional Interfaces
|
|
|
+we have seen the power of data abstraction, we will now look at a new
|
|
|
+powerful design principle for working with data structures:
|
|
|
+ conventional interfaces
|
|
|
+
|
|
|
+consider this procedure, analagous to count-leaves
|
|
|
+#+BEGIN_SRC scheme
|
|
|
+ (define (sum-odd-squares tree)
|
|
|
+ (cond ((null? tree) 0)
|
|
|
+ ((not (pair? tree))
|
|
|
+ (if (odd? tree) (square tree) 0))
|
|
|
+ (else (+ (sum-odd-squares (car tree))
|
|
|
+ (sum-odd-squares (cdr tree))))))
|
|
|
+#+END_SRC
|
|
|
+
|
|
|
+on the surface, it's rather different than this one for finding even
|
|
|
+fibs
|
|
|
+#+BEGIN_SRC scheme
|
|
|
+ (define (even-fibs n)
|
|
|
+ (define (next k)
|
|
|
+ (if (> k n)
|
|
|
+ (list)
|
|
|
+ (let ((f (fib k)))
|
|
|
+ (if (even? f)
|
|
|
+ (cons f (next (+ k 1)))
|
|
|
+ (next (+ x 1))))))
|
|
|
+ (next 0))
|
|
|
+#+END_SRC
|
|
|
+
|
|
|
+an abstract description shows a common nature
|
|
|
+
|
|
|
+program 1:
|
|
|
+ - enumerates the leaves of a tree
|
|
|
+ - filters, selecting odds
|
|
|
+ - squares selected ones
|
|
|
+ - accumulates the results with +, starting with 0
|
|
|
+
|
|
|
+program 2:
|
|
|
+ - enumerates integers from 0 to n
|
|
|
+ - computes fib for each int
|
|
|
+ - filters, selecting evens
|
|
|
+ - accumulates with cons, starting with empty list
|
|
|
+
|
|
|
+these follow a common pattern ging through the common stages (like a
|
|
|
+signal flow)
|
|
|
+ - enumerate
|
|
|
+ - filter
|
|
|
+ - map
|
|
|
+ - accumulate
|
|
|
+
|
|
|
+these programs could be more elegantly expressed by organizing them to
|
|
|
+follow a signal-flow idea
|
|
|
+
|
|
|
+SEQUENCE OPERATIONS
|
|
|
+
|
|
|
+concentrate on the "signals" that flow from one stage of the program
|
|
|
+to the next. representing these signals as lists allows us to perform
|
|
|
+list operations on them (LISt Processing ?!)
|
|
|
+
|
|
|
+to filter, accumulate, and enumerate
|
|
|
+#+BEGIN_SRC scheme :session signal
|
|
|
+ (define (filter predicate sequence)
|
|
|
+ (cond ((null? sequence) (list))
|
|
|
+ ((predicate (car sequence))
|
|
|
+ (cons (car sequence)
|
|
|
+ (filter predicate (cdr sequence))))
|
|
|
+ (else (filter predicate (cdr sequence)))))
|
|
|
+
|
|
|
+ (define (accumulate op initial sequence)
|
|
|
+ (if (null? sequence)
|
|
|
+ initial
|
|
|
+ (op (car sequence)
|
|
|
+ (accumulate op initial (cdr sequence)))))
|
|
|
+
|
|
|
+ ;; to enumerate even-fibs
|
|
|
+ (define (enumerate-interval low high)
|
|
|
+ (if (> low high)
|
|
|
+ (list)
|
|
|
+ (cons low (enumerate-interval (+ low 1) high))))
|
|
|
+
|
|
|
+ ;; to enumerate tree leaves
|
|
|
+ (define (enumerate-tree tree)
|
|
|
+ (cond ((null? tree) (list))
|
|
|
+ ((not (pair? tree)) (list tree))
|
|
|
+ (else (append (enumerate-tree (car tree))
|
|
|
+ (enumerate-tree (cdr tree))))))
|
|
|
+
|
|
|
+ (filter odd? (list 1 2 3 4 5))
|
|
|
+ (accumulate + 0 (list 1 2 3 4 5))
|
|
|
+ (enumerate-interval 2 7)
|
|
|
+ (enumerate-tree (list 1 (list 2 (list 3 4)) 5))
|
|
|
+#+END_SRC
|
|
|
+
|
|
|
+#+RESULTS:
|
|
|
+| 1 | 2 | 3 | 4 | 5 |
|
|
|
+
|
|
|
+with that, we can express those funcs as signal flows
|
|
|
+#+BEGIN_SRC scheme :session signal
|
|
|
+ (define (sum-odd-squares tree)
|
|
|
+ (accumulate +
|
|
|
+ 0
|
|
|
+ (map square
|
|
|
+ (filter odd?
|
|
|
+ (enumerate-tree tree)))))
|
|
|
+ (define (even-fibs n)
|
|
|
+ (accumulate cons
|
|
|
+ (list)
|
|
|
+ (filter even?
|
|
|
+ (map fib
|
|
|
+ (enumerate-interval 0 n)))))
|
|
|
+#+END_SRC
|
|
|
+
|
|
|
+this style encorages MODULARITY ... cool
|
|
|
+
|
|
|
+we can use those units for any other task, like a list of the squares
|
|
|
+of the first so many fib nums
|
|
|
+#+BEGIN_SRC scheme :session signal
|
|
|
+ (define (fib n)
|
|
|
+ (cond ((= n 0) 0)
|
|
|
+ ((= n 1) 1)
|
|
|
+ (else (+
|
|
|
+ (fib (- n 1))
|
|
|
+ (fib (- n 2))))))
|
|
|
+ (define (square x) (* x x))
|
|
|
+ (define (list-fib-squares n)
|
|
|
+ (accumulate cons
|
|
|
+ (list)
|
|
|
+ (map square
|
|
|
+ (map fib
|
|
|
+ (enumerate-interval 0 n)))))
|
|
|
+ (list-fib-squares 10)
|
|
|
+#+END_SRC
|
|
|
+
|
|
|
+#+RESULTS:
|
|
|
+| 0 | 1 | 1 | 4 | 9 | 25 | 64 | 169 | 441 | 1156 | 3025 |
|
|
|
+
|
|
|
+we could rearrange them to compute odd ints in a sequence
|
|
|
+#+BEGIN_SRC scheme :session signal
|
|
|
+ (define (product-of-squares-of-odd-elements sequence)
|
|
|
+ (accumulate *
|
|
|
+ 1
|
|
|
+ (map square
|
|
|
+ (filter odd? sequence))))
|
|
|
+ (product-of-squares-of-odd-elements (list 1 2 3 4 5))
|
|
|
+#+END_SRC
|
|
|
+
|
|
|
+#+RESULTS:
|
|
|
+: 225
|
|
|
+
|
|
|
+here's another example to get the highest programmer salary
|
|
|
+#+BEGIN_SRC scheme
|
|
|
+ (define (salary-of-highest-paid-programmer records)
|
|
|
+ (accumulate max
|
|
|
+ 0
|
|
|
+ (map salary
|
|
|
+ (filter programmer? records))))
|
|
|
+#+END_SRC
|
|
|
+
|
|
|
+this is just a hint of the great power conferred by sequence ops
|
|
|
+
|
|
|
+we will use em to enable infinite sequences in 3.5
|
|
|
+
|
|
|
+
|
|
|
+NESTED MAPPINGS
|
|
|
+
|
|
|
+the sequence paradigm can be used in the fashion of nested loops
|
|
|
+for this problem of pairs of numbers that add to primes
|
|
|
+#+BEGIN_SRC scheme
|
|
|
+ (accumulate append
|
|
|
+ (list)
|
|
|
+ (map (lambda (i)
|
|
|
+ (map (lambda (j) (list i j))
|
|
|
+ (enumerate-interval 1 (- i 1))))
|
|
|
+ (enumerate-interval 1 n)))
|
|
|
+
|
|
|
+ ;; this is very common, we can express it like so
|
|
|
+ (define (flatmap proc seq)
|
|
|
+ (accumulate append (list) (map proc seq)))
|
|
|
+
|
|
|
+ (define (prime-sum? pair)
|
|
|
+ (prime? (+ (car pair) (cadr pair))))
|
|
|
+ (define (make-pair-sum pair)
|
|
|
+ (list (car pair) (cadr pair) (+ (car pair) (cadr pair))))
|
|
|
+
|
|
|
+ (define (prime-sum-pairs n)
|
|
|
+ (map make-pair-sum
|
|
|
+ (filter prime-sum?
|
|
|
+ (flatmap
|
|
|
+ (lambda (i)
|
|
|
+ (map (lambda (j) (list i j))
|
|
|
+ (enumerate-interval 1 (- i 1))))
|
|
|
+ (enumerate-interval 1 n)))))
|
|
|
+#+END_SRC
|
|
|
+
|
|
|
+to generate unique permutations of a set, {1 2 3} {2 1 3} {3 1 2} etc
|
|
|
+#+BEGIN_SRC scheme
|
|
|
+ (define (permutations s)
|
|
|
+ (if (null? s)
|
|
|
+ (list (list))
|
|
|
+ (flatmap (lambda (x)
|
|
|
+ (map (lambda (p) (cons x p))
|
|
|
+ (permutations (remove x s))))
|
|
|
+ s)))
|
|
|
+ (define (remove item sequence)
|
|
|
+ (filter (lambda (x) (not (= x item)))
|
|
|
+ sequence))
|
|
|
+#+END_SRC
|