We get:
(defrel (alwayso)
(conde
(#s)
((alwayso))))
(run 1 q
(alwayso)
#u)
The book (2nd ed) says:
"
alwayso
succeeds, followed by#u
, which causes(alwayso)
to be retried, which succeeds again".
I still don't get the control flow. Why aren't both arms of the conde
tried (continuing in the recursion) before stepping out to #u
?
You meant, why the second arm is not tried before stepping out to
#u
with the result from the first.The short answer is, lazy-sequences (also, lazy-evaluation) paradigm.
Under the eager evaluation paradigm, each goal produces all if its solutions in full. If their number is unbounded ("infinite"), this means infinite looping, indeed as you expected.
Under the lazy evaluation paradigm, each goal produces its solutions one by one -- meaning, it produces only its first result and stays ready to produce the next when and if requested.
Goals produce their resulting substitutions one by one, as if by
yield
in Python. So do the combined goals as well. Theconde
combination combines its subgoal in a certain way (see below), and produces the combined goal's results one by one. Thusrun
's first goal, the combined goalconde
, produces its results one by one, each result being fed to the secondrun
goal,#u
, as soon as it is produced.If all the solutions of
run
's first subgoal were to be produced before feeding the list of results to the next subgoal, the evaluation would never end for any goal capable of producing infinite list (or more precisely, unbounded stream) of results.These streams are thus lazy streams, and lists are eager lists. The difference is operational. Under eager scheme, the first subgoal's list is first built in full, and only then processed by the next goal. When the number of results is infinite, building an infinite eager list will take infinite time.
Thus under the eager evaluation paradigm it would indeed get stuck in the recursion inside that
conde
.Under the lazy evaluation paradigm, chosen by the book's implementation, it gets stuck in a bigger loop, bouncing off the
#u
back every time. But theconde
works, producing its resulting substitutions successfully one by one.Scheme itself is an eager language. Delaying the production of the rest of stream is achieved by putting it behind the lambda, Roughly,
(an eager list) is replaced with
(a lazy stream).
alwayso
is defined so that it produces an infinite stream of copies of its input substitution, unchanged. But it produces them one by one.Then
run
feeds this stream from its first goal, to the second goal,#u
, which rejects it. Sincerun 1
demands one solution from its subgoals, it retries them until one solution / substitution goes through.Which never happens.
So this should result in infinite looping.
Again, both arms are tried -- first, the first one; then, after its (one) result gets rejected by the subsequent
#u
, the second arm is tried. And the resulting substitution gets rejected, again. Ad infinitum:Getting stuck.
Following the implementation's definitions closer,
so indeed we see here a definition which produces a stream of an unbounded number of copies of the input substitution
s
, as and when requested, as the result of a call((alwayso) <s>)
.Or, in pseudocode, writing
++
forappend-inf
and[s]
for(list s)
,Finally,
and no matter how many
empty-s
's it applies the(^s '())
to, to append the results together, all the results are empty lists, so it can'ttake
even one of the contained results because there are none. In pseudocode, writings0
forempty-s
,Or, symbolically,
So it gets stuck.