(ns verbal-arithmetic
(:require
[clojure.core.logic :refer [all run* everyg lvar == membero fresh conde succeed fail conso resto]]
[clojure.core.logic.fd :as fd]))
(comment
"Solving cryptarithmetic puzzle"
" SEND
+ MORE
______
MONEY")
(defn send-more-money-solutions []
(run* [s e n d m o r y]
(fd/in s e n d m o r y (fd/interval 0 9))
(fd/!= s 0)
(fd/!= m 0)
(fd/distinct [s e n d m o r y])
(fd/eq (= (apply + [(* 1000 s) (* 100 e) (* 10 n) d
(* 1000 m) (* 100 o) (* 10 r) e])
(apply + [(* 10000 m) (* 1000 o) (* 100 n) (* 10 e) y])))))
The above example doesn't work because apply
does not work correctly in fd/eq
. The following version of send-more-money-solutions
works because I don't use apply
. I need to use apply
to generalize the solution to work with arbitrary strings with different length.
(defn send-more-money-solutions []
(run* [s e n d m o r y]
(fd/in s e n d m o r y (fd/interval 0 9))
(fd/!= s 0)
(fd/!= m 0)
(fd/distinct [s e n d m o r y])
(fd/eq (= (+ (* 1000 s) (* 100 e) (* 10 n) d
(* 1000 m) (* 100 o) (* 10 r) e)
(+ (* 10000 m) (* 1000 o) (* 100 n) (* 10 e) y)))))
What should I do? (For above, I have an idea that I could maybe write a macro (although not sure how yet) but actually I need to be able to use variables that is a sequence of logic variables. Something like below)
(fd/eq (= (+ (apply + lvars1) (apply + lvars2))
(apply + lvars3)))
The error message looks like
java.lang.IllegalArgumentException: Can't call nil, form: (nil + [(* 1000 s) (* 100 e) (* 10 n) d (* 1000 m) (* 100 o) (* 10 r) e] G__1124704)
I think something weird is going on in fd/eq
macro so I should try without using eq
macro.
Thank you all in advance!
Exactly, a general solution to this problem is to introduce an arbitrary, dynamic number of logic variables and relate/constrain them.
Solver
First define some recursive goals to work with sequences of logic variables. (Luckily I already had these around for previous problems!)
Relate the sum of a sequence of logic variables to another logic variable:
Relate the sum of products of two sequence of logic variables to another logic variable:
Plus a little helper function to generate the magnitude multipliers:
Then wire it all together:
Some of this should look familiar from your example, but the major differences are that the number of words (and their characters) can be handled dynamically. Fresh logic variables are created with
lvar
for the set of all characters, as well as the sums for each word. Then the logic variables are constrained/related usingeveryg
and the recursive goals above.Sample Problems
The function will return all solutions for the given words, and "send more money" only has one possible solution:
Another example with four words is "cp is fun true" (see Google Cryptarithmetic Puzzles) which has 72 possible solutions:
This is the biggest one I could find is on Wikipedia, and the function finds the only solution in ~30s on my laptop:
And here's a function to pretty print the results: