|  | @@ -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
 |