« Back to James's CS61A page. | CS61A: Spring 2000 |
(The long list of things I planned to talk about but was too tired to go through. Sorry!)
Now that we have mutable data, we have a lot more flexibility to manipulate lists.
It turns out that the list is a very flexible data structure. We can use a list to create a lot of other structures. (See the foreword to the SICP textbook, top of page xiv.)
Although data structures are the topic of CS61B, I think it's important to be introduced to several of them now. Two of the most common data structures used in computer science are the queue and the stack.
A queue is a data structure that stores some arbitrary items. Items in a queue are like people waiting in line; the items are handled on a first-come, first-serve basis. Queues are thus a FIFO (first in, first out) buffer.
Following the analogy to a line of people, the ends of a queue are known as the front and the rear. Items can only be added to the rear and can only be removed from the front.
Adding an item to the rear of a queue often is called enqueuing, and removing the front item is called dequeuing.
Queues are discussed and implemented in section 3.3.2 of the textbook.
In contrast to queues are stacks. Items in a stack are like dinner plates arranged one on top of another; a plate can only be added to or be removed from the top. Stacks are thus a LIFO (last in, first out) buffer.
Following the analogy to a physical stack of plates, the ends of a stack are known as the top and the bottom.
Adding an item to the top of a stack is known as as pushing, and
removing the
Exercise: Implementing stacks
Why do we need *table* as the first element in the implementation for tables?
First, as we already know, when we deal with many different kinds of
abstract data types, it's often a good idea to use
This, however, isn't a complete answer, because in the case of tables, the *table* header really isn't optional. Why not?
(If you've completed my stack exercise, hopefully you already should understand the need for dummy headers. If you haven't done the exercise yet, go do it!)
The problem is that set! cannot mutate structures. set! can only modify bindings. To exemplify this, let's try to add an item to the beginning of a list without using dummy headers:
(define (insert-front! lst item) (set! lst (cons item lst)) ) ;; WRONG
To add an item from the front of a list, we must change the initial pointer to the list. Is this possible with set!? Using the above definitions, will the following code work as intended?
(define meta-vars (list 'foo 'bar 'baz)) (insert-front! meta-vars 'qux) ;; WRONG ;; this does not mutate meta-vars!
The problem is that
This is a common problem when dealing with programming languages that use
(Note that when we invoke a procedure with a pair structure as an argument, a copy of the initial pointer is made, not a copy of the structure!)
The common workaround to this is to use an extra level of indirection. This means that we create an extra pointer to the item we really want to modify. In the cases of mutable data structures in Scheme, we create an extra, dummy cons pair to accomplish this.
Memoization (not memorization, but the same idea) is the process of saving return values so we don't have to recompute them later. This is very useful when we need to perform expensive calculations repeatedly. Often the cost of of storing a value and looking it up again later is much less than recomputing the value from scratch.
Keep in mind, however, that memoization only saves return values. Consider what happens when we introduce side-effects!
First, here's a more general version of memoize that handles functions with any number of arguments:
(define (memoize f) (let ((table (make-table))) (lambda args ;; lambda equivalent of dotted-tail notation (let ((previously-computed-result (lookup args table))) (or previously-computed-result (let ((result (apply f args))) (insert! args result table) result) )) )) )
Now, for some examples with side-effects:
(define x 0) (define count (lambda () (set! x (+ x 1)) x) ) (define memo-count (memoize (lambda () (set! x (+ x 1)) x) )) (define hello (lambda () (display "Hello, world!") (newline) 'okay) ) (define memo-hello (memoize (lambda () (display "Hello, world!") (newline) 'okay) ))
What happens when we first call
« Week 8 | Week 10 » |
« Back to James's CS61A page. | CS61A: Spring 2000 |
Last Modified: Tuesday, 30-Dec-2014 11:58:34 PST