💾 Archived View for gnebbia.net › notes › sicp › sections › 2_substitution_model.gmi captured on 2024-12-17 at 09:43:47. Gemini links have been rewritten to link to archived content

View Raw

More Information

⬅️ Previous capture (2023-01-29)

-=-=-=-=-=-=-

Procedures and Processes

In this section we analyze how procedures that we write affect processes

in a machine. We learn about linear iteration, linear recursion and

tree recursion processes. All these processes are all built using

recursive procedures since in Lisp this is very natural and used as a

common approach.

In order to understand how procedures affect processes we use the

until we introduce a more complex model to understand how procedures

are managed within a machine.

This model can be thought of as the classical mechanics or ideality in

physics, when we are ok with a simplistic approximation. Anyway for some

other scenarios we will find out it is not enough to use the substitution

model and will be forced to deal with a more complex model.

Kinds of Expressions

Procedures are made of expressions, which can be of these types:

- Numbers

- Symbols

- Combinations

- Lambda Expressions (special form)

- Definitions (special form)

- Conditionals (special form)

Note that "special form" means that for that kind of expression

we have other specific rules.

Substitution Model Rule

Once we have a procedure and a procedure call we can understand what

processes are generated by this procedure call by using the so called

"substitution model".

The steps to evaluate an application and observe the related generated

process can be described with the following steps:

- evaluate the operator to get the procedure;

- evaluate the operands to get the arguments;

- apply the procedure to the arguments:

- if the procedure is compound (not primitive) copy the body of the

procedure substituting the arguments supplied for the formal

parameters of the procedure;

- evaluate the resulting body;

Example of Substitution Model Rule

Let's apply the substitution rule to the following

simple procedure.

(define (+ x y)
    (if (= x 0)
        y
        (+ (dec x) (inc y))))

Now let's say the procedure gets called with:

(+ 3 4)

Now to apply the substitution model we can just do:

(+ 3 4)
(if (= 3 0) 4 (+ (dec 3) (inc 4)))
(+ (dec 3) (inc 4))
(+ (inc 3) 5)
(+ 2 5)
(if (= 2 0) 5 (+ (dec 2) (inc 5)))
(+ (dec 2) (inc 5))
(+ (dec 2) 6)
(+ 1 6)
(if (= 1 0) 6 (+ (dec 1) (inc 6)))
(+ (dec 1) (inc 6))
(+ (dec 1) 7)
(+ 0 7)
(if (= 0 0) 7 (+ (dec 0) (inc 7)))
7

Substitution Model Variants: Applicative Order vs Normal Order

Our method of evaluation by evaluating operator, evaluating the

operands and then applying the operator is just one possible rule of

evaluation. The ordering we have been using is called "Applicative

Order". An alternative method of evaluation would be to not evaluate

the operand until the value is needed. This method is called "Normal

Order". We can see the difference between these 2 from the following

example:

(square (+ 3 2))

Note that the input to square is (+ 3 2).

Applicative Order:

(square (+ 3 2))
(square 5)
(* 5 5)
25

In Applicative Order, you would evaluate the parameter x, before you

go the body of square, which is `(* x x)`. When you evaluate (+ 3 2),

you get 5 and this is what you pass into square. So x is bound to 5.

Normal Order:

(square (+ 3 2))
(* (+ 3 2) (+ 3 2))
(* 5 5)
25

In Normal Order, you don't evaluate (+ 3 2) until you absolutely need

to. So in this case, the x in (square x) is bound to (+ 3 2).

Notice that, in Normal Order, since you don't evaluate the x, which is (+

3 2), until it's needed, you need to evaluate it twice. On the other hand,

in Applicative Order, since you evaluate the operand x before applying

the procedure, you only evaluate (+ 3 2) once.

Procedures vs Processes

In this subsection we analyze two procedures which are functionally

identic, but produce a quite different process.

This is imporant because allow us to understand the relation

between a procedure and process and how to evaluate

the result of an applied substitution model.

Peano Arithmetic Sum v1 (Iterative Process)

```scheme

(define (+ x y)

(if (= x 0)

y

(+ (dec x) (inc y))))


Its substitution model for the procedure call `(+ 3 4)` is:

(+ 3 4)

(+ 2 5)

(+ 1 6)

(+ 0 7)

7



# Peano Arithmetic Sum v2 (Recursive Process)
 ```scheme
(define (+ x y)
    (if (= x 0)
        y
        (inc (+ (dec x) y))))

Its substitution model for the procedure call `(+ 3 4)` is:

(+ 3 4)
(inc (+ 2 4))
(inc (inc (+ 1 4)))
(inc (inc (inc (+ 0 4))))
(inc (inc (inc (4))))
(inc (inc 5))
(inc 6)
7

Conclusion Notes on Sum v1 Substitution Model

These processes have a very different shapes.

Note that the shape of the process is 2d and each dimension has a meaning.

The y axis is time, while the x axis is the space, so amount of things

I have to remember.

This v1 version has a time complexity of O(x) since the number of steps

on the y axis increases linearly with increasing x.

But has a constant complexity in space O(1).

A process which has a linear time complexity and a constant space complexity

is called **iteration process** (linear iteration).

These complexities put a bound, saying that every machine can perform

this procedure with this amount of resources at maximum.

Note that of course we are abstracting details such as how big is a number,

but it doesn't matter since we are at a different level of abstraction.

We could also consider that all the numbers take the same space.

Conclusion Notes on Sum v2 Substitution Model

This v2 version has a time complexity of O(x) since the number of steps

on the y axis increases linearly with increasing x.

And the same happens for space, if we increase x we have two more

lines, hence space complexity is also O(x).

A process which has both linear time complexity and space complexity

is called **recursion process** (or linear recursion).

Tree Recursion

This happens everytime we have a recursive process where two or more

calls to the same function are performed. As an example the fibonacci

sum can be computed very naturally in this way. Although an iterative

version would be more efficient, the tree recursive way is more natural

to design and is very easy to think in recursive terms. Other functions

using tree recursion are functions operating on complex data structures

such as binary trees, b trees, graphs and so on, which have a huge amount

of applications in computer science.

Additional Notes

Both iteration and recursion process have been defined using a recursive

definition (with recursive functions), since we call the function itself

in the studied function.

There is nothing special about this, we just have a recursive function

leading to two different process shapes (which have different names).

So the takeaway here is, expressed by the following note.

NOTE: we must be careful not to confuse the notion of a **recursive process**

with the notion of a **recursive procedure**.

In addition to sum up, in this section we have seen the relationship

between procedures and processes, and how to apply the substitution

model to generate the resulting process starting from a procedure and

a procedure call. We have then observed and discussed common different

processes, namely:

- iterative processes;

- linear recursive processes;

- tree recursive processes;