Chapter 3: Modularity, Objects, and State
so far we have reviewed the primitive elements of how programs are made and how those primitive elements can be combined in complex ways.
we learned the value of abstraction and modular programs.
the nature of a program thot it can be divided "naturally" into coherent parts that can be separately developed and maintained
we now look towards how to design entire systems. we will look at two prominent strategies for program construction:
seeing a program as a collection of distinct objects whose behaviors may change over time
seeing a program as a flow of information in a system
we will finally discard the substitution model of chapter 1 in favor of the environment model
we ordinarily view the world as a collection of independent objects each of which has state that changes over time.
at this point (in ch 3!!!) we will discuss local state variables and the assignment operator
we now have a comptational object with time-varying state
this means:
(withdraw 25) --> 75 (withdraw 25) --> 25 (withdraw 60) --> "Insufficient Funds" (withdraw 15) --> 35
these are not pure funtions
when evaluated twice, they yield different results
this is new before, we computed mathematical functions
(define balance 100) (define (withdraw amount) (if (>= balance amount) (begin (set! balance (- balance amount)) balance) "Insufficient Funds"))
we have the new language features, set! which sets a new value to a variable:
(set! name new-value)
and we have the (begin … … …) form, which returns the value of its final expression
there's some trouble with that begin variable. we can make it internal by encapsulating it:
(define new-withdraw (let ((balance 100)) (lambda (amount) (if (>= balance amount) (begin (set! balance (- balance amount)) balance) "Insufficient Funds"))))
combining set! with local variables is the general technique for constructing computational objects.
–> here we have just annhialated our substitution model. according to it, the above is incomprehensible. we will address later. first, a few variations on this new and very exciting theme
(define (make-withdraw balance) (lambda (amount) (if (>= balance amount) (begin (set! balance (- balance amount)) balance) "Insufficient Funds")))
make-withdraw
which can be used to make accounts with custom amounts
(define W1 (make-withdraw 100)) (define W2 (make-withdraw 100)) (W1 50) (W2 70)
30
these two withdrawal accounts are independent of each other
to make a functional bank account:
(define (make-account balance) (define (withdraw amount) (if (>= balance amount) (begin (set! balance (- balance amount)) balance) "Insufficient Funds")) (define (deposit amount) (set! balance (+ balance amount)) balance) (define (dispatch m) (cond ((eq? m 'withdraw) withdraw) ((eq? m 'deposit) deposit) (else (error "Unknown request -- MAKE-ACCOUNT" m)))) dispatch)
make-account
and to use it:
(define acc (make-account 100)) ((acc 'withdraw) 25) ((acc 'withdraw) 80) ((acc 'deposit) 120) ((acc 'withdraw) 60)
135
and:
(define acc2 (make-account 100))
acc2
will be a completely different account
with assignment, we can encapsulate state, rather than injecting it via formal parameters all over the place.
set! is cool, but it breaks the substitution model.
it breaks all "nice" mathematical models.
the reason it doesnt work is variable scope. like closures, i think. or like the kind of closure this book said is the wrong thing.
as we did in the first two chapters of the book, to not use assignment
makes extensive use of assignment
SAMENESS AND CHANGE
"equals can be substituted for equals" without changing the value of an expression
set! violates referential transparency.