In 85 there is:
(run* (x y)
(teacupo x)
(teacupo y))
which expands into:
(run* (x y)
(disj 2
(== 'tea x)
(== 'cup x))
(disj 2
(== 'tea y)
(== 'cup y)))
So then how does conj2
accumulate the results to be ((tea tea) (tea cup) (cup tea) (cup cup))
? I don't think this has been explained properly in the book.
The first goal in the conjunction instantiates the
x
, the second instantiates they
, andrun (x y)
says to report both.Both
x
andy
get instantiated becausex
is different fromy
, so they are independent from each other.This can be modeled under the solutions streams' combination paradigm as in this answer of mine (equivalent to the book's model sans the logical variables), as this pseudocode:
(1.)
s
is a solution, a substitution, a record of(name,value)
pairs. It is extended by unification:(all the operators like
++:
("append" a.k.a. "mplus"),||:
("or" a.k.a. "disj2"),>>:
("push-through" a.k.a. "bind"),&&:
("and" a.k.a. "conj2"), etc. are defined in the linked answer. The syntax is a free-licensed "curried application by juxtaposition", similar to Haskell's).A substitution represents our knowledge so far. The unification
(x ≡ TEA)
, say, is a goal -- a procedure which, given a substitution, produces a stream of new substitutions (here, one or none), augmented with the new knowledge thatx ≡ TEA
(we use all-caps names as auto-quoted symbols).Augmenting a substitution involves possible rejection of irreconcilable values.
The result is a list containing either one valid, possibly extended substitution, or an empty list containing none, meaning
(x,v)
was incompatible withs
. This expresses the notion that a logical variable's value, once set, can not change (except through backtracking).(2.) A disjunction involves application of the goals, separately, and passing along all the resulting valid solutions (substitutions) together:
(3.) A conjunction passes each solution of the first goal through the second:
All in all, four solutions, four substitutions satisfying the goal
teacup° x &&: teacup° y
, were produced.A conjunction
&&:
of two goals will, for each of the solutions produced by the first goal, collect all the solutions produced by the second which are conformant with the solution received by the second goal. Here, sincex
andy
are two separate variables, every new solution does not collide with the old, in other words, none of the solutions are rejected.(4.) If we'd used the same variable twice, their interaction would mean that we must reject irreconcilable substitutions:
Here, two solutions are produced because the same logical variable can not hold two different values at once, so e.g.
(x ≡ CUP) [(x,TEA)]
resolves as[]
, the empty list containing no solutions, thus rejecting the attempted assignment(x,CUP)
when(x,TEA)
is already present.(5.)
(run* (x y) (teacup° x) (teacup° y))
is more or less likefor a suitably defined
report <vars>
goal.(6.) And
run* (x) (teacup° x) (teacup° x))
is more or less like