-
Notifications
You must be signed in to change notification settings - Fork 126
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Book error: Exercise 3.57 is incorrect, memoization of add_streams is insufficient #1063
Comments
Debug codeI'm providing the following debugging code that I used. There are some syntax elements that are from standard JS and not Source, but it should be fairly straightforward to execute in a local NodeJS environment with the sicp npm package. import { pair, is_null } from 'sicp';
import { head, stream_tail } from 'sicp';
import { stream_ref } from 'sicp';
const ITER = 25;
const OPT_ITER = 25;
const MEMO_ITER = 50;
function memo(fun) {
let already_run = false;
let result = undefined;
return () => {
if (!already_run) {
result = fun();
already_run = true;
return result;
} else {
//console.log("returning stored");
return result;
}
};
}
(function unoptimized() {
function stream_map_2(f, s1, s2) {
return is_null(s1) || is_null(s2)
? null
: pair(f(head(s1), head(s2)),
() => stream_map_2(f, stream_tail(s1), stream_tail(s2)));
}
let addCalls = 0;
function add_streams(s1, s2) {
const add = (a, b) => {
addCalls++;
return a + b;
};
return stream_map_2(add, s1, s2);
}
const fibs = pair(0,
() => pair(1, () => add_streams(stream_tail(fibs), fibs)));
console.log("Unoptimized:");
for (let i = 0; i < ITER; i++) {
addCalls = 0;
console.log(`Fib(${i+1}): ${stream_ref(fibs, i)}\tAdd calls: ${addCalls}`);
}
console.log();
})();
(function optimized() {
function stream_map_2_optimized(f, s1, s2) {
return is_null(s1) || is_null(s2)
? null
: pair(f(head(s1), head(s2)),
memo(() => stream_map_2_optimized(f, stream_tail(s1), stream_tail(s2))));
}
let addCalls = 0;
function add_streams(s1, s2) {
const add = (a, b) => {
addCalls++;
return a + b;
};
return stream_map_2_optimized(add, s1, s2);
}
// "optimized" fibs; doesn't correctly memoize all stream constructions
const fibs = pair(0,
() => pair(1, () => add_streams(stream_tail(fibs), fibs)));
/* correctly memoized version - uncomment this section to use
const fibs = pair(0,
memo(() => pair(1, memo(() => add_streams(stream_tail(fibs), fibs)))));
*/
console.log("Optimized:");
for (let i = 0; i < OPT_ITER; i++) {
addCalls = 0;
console.log(`Fib(${i+1}): ${stream_ref(fibs, i)}\tAdd calls: ${addCalls}`);
}
console.log();
})();
(function all_streams_memoized() {
function cons_stream(sHead, sTailFunc) {
return pair(sHead, memo(sTailFunc));
}
function stream_map_2(f, s1, s2) {
return is_null(s1) || is_null(s2)
? null
: cons_stream(
f(head(s1), head(s2)),
() => stream_map_2(f, stream_tail(s1), stream_tail(s2)));
}
let addCalls = 0;
function add_streams(s1, s2) {
const add = (a, b) => {
addCalls++;
return a + b;
};
return stream_map_2(add, s1, s2);
}
const fibs = cons_stream(0,
() => cons_stream(1, () => add_streams(stream_tail(fibs), fibs)));
console.log("With cons_stream:");
for (let i = 0; i < MEMO_ITER; i++) {
addCalls = 0;
console.log(`Fib(${i+1}): ${stream_ref(fibs, i)}\tAdd calls: ${addCalls}`);
}
console.log();
})(); |
Looking at your first post, you write: However, when I actually tried it I found that the number of addition operations is actually identical. In fact, the memoized functions are never called. Well, the optimization will be significant, when you use the result stream. Here is an example: So I don't think this is a bug in SICP JS. |
You are rightly pointing out that SICP JS 3.5 significantly deviates from SICP 3.5. The reason for this is that JavaScript does not have macros. In Scheme, the function |
Yes, However, the bug that I am pointing out is not with I've added on to your example to demonstrate:
Makes sense, thank you for the explanation. Hopefully I'll understand this deeper once I've finished the book. |
I see. Thanks for your patience explaining. Indeed, the problem is that the two tails in fibs are not memoized. Your solution is correct. So the bug lies in the formulation of Exercise 3.57. It should read as follows (with emphasis added): Exercise 3.57 How many additions are performed when we compute the nth Fibonacci number using the declaration of fibs based on the add_streams function? Show that this number is exponentially greater than the number of additions performed if add_streams had used the function stream_map_2_optimized described in exercise 3.50, and if we had declared fibs like this: const fibs = pair(0, |
Indeed: A |
I believe I have found an error in the book.
The Error (Exercise 3.57)
Exercise 3.57 is defined as:
This implies that the following would only require a linear number of additions:
However, when I actually tried it I found that the number of addition operations is actually identical. In fact, the memoized functions are never called. There's a whole bunch of state and stored functions that make reasoning about it extremely convoluted even for small indices, but the root cause is that in the definition of fibs, we are not fully memoizing the stream tail functions. The following fix correctly runs with a linear number of additions:
This correctly memoizes all stream_tail functions.
Comparison to 1985 original
I checked the original Scheme version, which does not have this defect. Comparing to the original Scheme version, Chapters 3.5.1 and 3.5.2 are some of the most heavily edited: https://sicp.sourceacademy.org/chapters/3.5.1.html, and in particular the JS version gets rid of
cons-stream
, instead usiong rawpair
operations for all stream construction.The Scheme version of exercise 3.5.7 is:
The question is in fact sort of reversed; the implementation of
fibs
is optimal, and the exercise suggests thinking of a potential unoptimized version – namely, not using memoization. This works because the memoization is baked into thecons-stream
procedure, which becomes the primitive constructor for all streams.Given the functions already available, a JS implementation would be something like this:
By using this constructor everywhere that we make streams with
pair
currently, the optimization becomes "baked in"; in fact,stream_map_2
can automatically take advantage of it without explicitly callingmemo
:The definition of
fibs
here is pretty much identical to the one given for Scheme:Potential fixes
Short term
The exercise 3.57 as presented is mistaken, so it should be fixed. Here is a sample rewritten version:
Long term (2nd edition?)
I don't know the reasons that the authors decided to do away with
cons-stream
in the JavaScript edition, but in my comparison this feels like a change that could be reverted. One strong message of the book as a whole is to use constructors and selectors as abstractions for the implementation details, so havingcon_stream
as a constructor which wrapspair(sHead, memo(sTailFunc))
seems reasonable.I did run into one problem when trying to fully lean into copying the Scheme implementation. Namely, I tried to define
cons_stream
as follows:I could not longer create streams that referenced themselves:
I don't know if this is a difference between JS and Scheme on how expressions are treated, and I confess that my understanding is not deep enough to definitively show it one way or another. Skimming future sections, I see more material on
delay
, normal vs applicative order etc. so perhaps there is an explicit reason for changing this implementation.However from what I can tell, there would be no drawback by providing the memoized implementation of
cons_stream
, where the two arguments must be a stream head, and a function returning the stream tail.The text was updated successfully, but these errors were encountered: