ch 1 recap:
used primitive data (numbers) and primitive operations (arithmetic)
combined procedures to form compound procedures through
composition
conditionals
use of parameters
saw that proocedures can be regarded as a pattern for local evolution of a process
classified, reasoned about, performed simple algorithmic analyses of common patterns for processes embodied in procedures
saw thath higher-order procedures empower us by allawing us to manipulate and reason about general methods of computation
this is much of the essence of programming
now, we will approach complex data.
programs typically model complex phenomena, and often, "we construct computational objocts that have seweral parts in ordor to model real-world phenomena that have several aspects"
in ch 1 we made compound procedures. in ch2 we will use compound data
neat!
coming up:
data "glued together"
isolating the parts of a program that deal with how data is represented from the parts that deal with how data is used
shielding a process from the more primitive details it deals with
!! the distinction between data and procedure continues to blur at a breakneck pace
the glue for combining data works with primitive as well as compound data
combining program modules in a mix-n-match fashion with compound data
data whose elements can be arbitrary symbols
operations that can handle many kinds of data
a technique that allows individual data representations to be combined in isolation and combined additively
combining data representations without modification
isolating how a compound data object is used from the details of how it's constructed
the interface between a concrete data object and its abstract operations
suppose…we want to do arithmetic with rational numbers
we want to add, subtract, multiply, divide, and test for equality
let's assume the selectors and constructors are available as procedures:
returns rational number with numerator n and denominator d
returns numerator of rational number x
returns denominator of rational number x
we can happily use wishful thinking here. these procedures do not exist, but we can pretend they do and build a suite of funtionality:
(define (add-rat x y) (make-rat (+ (* (numer x) (denom y)) (* (numer y) (denom x))) (* (denom x) (denom y)))) (define (sub-rat x y) (make-rat (- (* (numer x) (denom y)) (* (numer y) (denom x))) (* (denom x) (denom y)))) (define (mul-rat x y) (make-rat (* (numer x) (numer y)) (* (denom x) (denom y)))) (define (div-rat x y) (make-rat (* (numer x) (denom y)) (* (denom x) (numer y)))) (define (equal-rat? x y) (= (* (numer x) (denom y)) (* (numer y) (denom x))))
we have operations based on the selector and constructor procedures
numer
denom
make-rat
but these don't exist yet. we need a way to glue together two numbers to make a rational number
a compound structure provided by lisp
pairs can be constructed with the primitive procedure cons
given a pair, we can extract the elements with car
and cdr
(define x (cons 1 2)) (car x)
1
(cdr x)
2
–> a note on terminology
cons
is short for "construct"
the other two harken back to the IBM 704 implementation of lisp. the machcine had an addressing scheme that had references to the "address" and "decrement" parts of a memory location.
car
is short for "Contents of Address part of Register"
cdr
is short for "Contents of Decrement part of Register"
cdr is pronounced "could-er"
(define x (cons 1 2)) (define y (cons 3 4)) (define z (cons x y)) (car (car z)) ; 1 (car (cdr z)) ; 3
!! the single compound primitive pair, implemented by the procedures
cons
, car
, cdr
, is the only glue we need
data objects built by pairs
(define (make-rat n d) (cons n d)) (define (numer x) (car x)) (define (denom x) (cdr x)) (define (print-rat x) (newline) (display (numer x)) (display "/") (display (denom x)))
including a tostr method, we can use our new procedures for great fun
(define one-half (make-rat 1 2)) (print-rat one-half) (define one-third (make-rat 1 3)) (print-rat (add-rat one-half one-third))
this does not reduce the terms. we can do this if we have a gcd like back in 1.2.5
(define (make-rat n d) (let ((g (gcd n d))) (cons (/ n g) (/ d g))))
there are abstraction barriers here
——|programs that use rational nums|——-
rational numbers in problem domain
————|add-rat sub-rat … |————
rational numbers as numers and denoms
————|make-rat numer denom|————
rational numbers as pairs
—————-|cons car cdr|—————-
however pairs are implemented
these abstraction barriers isolate the various levels of the system. at each level, the barrier separates the programs above from the programs below that implement the abstractions. the procedures at each level are the interfaces that define the abstraction barriers and connect the different levels.
changing from one representation to another should have minimal (none??!!) effects on the levels above.
what is data, man?
considering make-rat, for any int n and any nonzero int d, if x is (make-rat n d), this is true:
(numer x) n ----------- = --- (denom x) d
data is defined by some collection of selectors and constructors with specified conditions that the procedures must fulfil in order to be valid.
this pount of view can define high level data objects such as rats but also low level objects like pairs.
the condition pair operations satisfy:
if z is (cons x y)
(car z) is x
(cdr z) is y
we could implement pairs with no data structures, only procedures
(define (cons x y) (define (dispatch m) (cond ((= m 0) x) ((= m 1) y) (else (error "argument not 0 or 1 -- CONS" m)))) dispatch) (define (car z) (z 0)) (define (cdr z) (z 1))
this is an obscure but valid way to implement pairs.
"procedural representations of data will play a central role in our programming repertoire."
this style of programming
will visit this in ch 3 re the issues of modeling and simulation
Alyssa P. Hacker is building some cool shit
the system will give ability to manipulate inexact quantites with known precision, such as with resistors.
calculate parallel resistance:
1 Rp = ------------- 1/R1 + 1/R2
alyssa postulates the existance of an abstract obj known as an "interval" with an upper and lower bound
she presumes that given the endpoints of an interval, she can construct an interval with make-interval
she writes a procedure for adding two intervals:
(define (add-interval x y) (make-interval (+ (lower-bound x) (lower-bound y)) (+ (upper-bound x) (upper-bound y))))
she also has a procedure for the product:
(define (mul-interval x y) (let ((p1 (* (lower-bound x) (lower-bound y))) (p2 (* (lower-bound x) (upper-bound y))) (p3 (* (upper-bound x) (lower-bound y))) (p4 (* (upper-bound x) (upper-bound y)))) (make-interval (min p1 p2 p3 p4) (max p1 p2 p3 p4))))
and the reciprocal, for division
(define (div-interval x y) (mul-interval x (make-interval (/ 1.0 (upper-bound y)) (/ 1.0 (lower-bound y)))))