b=(a+2)*(a+1) c=a*b a=3Choice 1:
c=(3)*(3+2)*(3+1)Choice 2:
c=3*5*4
c=60
c= (3)*((3+2)*(3+1))
c=(3)*(5*4)
c=(3)*(20)
c=60
They look very similar and yield the same result, however they act differently. In choice 1, the numbers are substituted and evaluated at the end. In Choice 2, the intermediate step (solving for b) is completed before evaluating for c. Choice 2 is applicative order, or strict evaluation. Choice 1 is normal order, also known as lazy evaluation because it only calls what is needed. The Lisp interpreter acts like Choice 2.
Knowing that lisp is applicative-order, we can now theorize the results of the somewhat famous Ben Bitdiddle Test, exercise 1.5 in SICP.I attempted to predict the outcome yesterday when I did not have access to my computer. I'm still bitter about it. First is the example written in lisp. Second is the test translated into Python. Do you think lisp and python act the same? What will they do when tested?
(define (p) (p))Or in Python
(define (test x y)
(if (= x 0)
0
y))
(test 0 (p))
def p():
return p()
def test(x, y):
return 0 if x==0 else y
test(0, p())
I spent about 45 minutes irritated, thinking there was no way lisp would return either 0 or y, because the p would be looping forever, I was able to get on a computer. I WAS RIGHT! Applicative order interpreters cannot evaluate the test, because they are too concerned with the definition of p. "Lazy" or normal order interpreters only apply the needed information, so they are able to return 0. According to this test, Python is also a strict interpreter, using applicative order like lisp. I expect the differences between normal order and applicative order will be relevant again in further reading of SICP.
I googled for applicative order, and your blog contained exactly the code snippet I needed, so I included a link to this post on my blog.
ReplyDeleteThanks!