-
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmodels.py
128 lines (109 loc) · 3.63 KB
/
models.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
from typing import List # , Int # , Dict, Tuple
import sympy as smp
# Global Solvit Symbols
A, B, C, D, E, F, G, H, J = smp.symbols("A,B,C,D,E,F,G,H,J")
GVM = {
"A": A,
"B": B,
"C": C,
"D": D,
"E": E,
"F": F,
"G": G,
"H": H,
"J": J,
}
# smp.solve((f1,f2,f3,f4,f5,f6,f7,f8),(A,B,C,D,E,F,G,H,J))
# smp.solve([e.fx for e in self.expressions],list(GVM.values()))
class Expression:
def __init__(self, variables: List, total: int):
self.variables = [v.upper() for v in variables]
self.total = total
@property
def fx(self):
return smp.Eq(sum([GVM[k] for k in self.variables]), self.total)
def __repr__(self):
return "+".join(self.variables) + "=" + str(self.total)
def reduce_expr(x1: Expression, x2: Expression):
i = set(tuple(x1.variables)).difference(set(tuple(x2.variables)))
j = set(tuple(x2.variables)).difference(set(tuple(x1.variables)))
x, y = list(i), list(j)
d = x1.total - x2.total
print("+".join(x), " = ", "+".join(y), " ", d)
class Puzzle:
MainExpr = smp.Eq(sum(GVM.values()), 45)
Symbols = list(GVM.values())
def __init__(self, expressions: List[Expression]):
self.expressions = expressions
self.solution = {
"A": 0,
"B": 0,
"C": 0,
"D": 0,
"E": 0,
"F": 0,
"G": 0,
"H": 0,
"J": 0,
}
def __repr__(self):
return str(
"\n".join(
["+".join(e.variables) + "=" + str(e.total)
for e in self.expressions]
)
)
def var_intersects(self, *args):
if args:
v = self.var_expressions(args[0])
for a in args:
v = v.intersection(self.var_expressions(a))
return v
def var_expressions(self, var):
var = var.upper()
assert var in self.solution.keys()
exps = set([e for e in self.expressions if var in e.variables])
return exps
def reduce_exprs(self):
rform = smp.solve([e.fx for e in self.expressions] + [Puzzle.MainExpr])
# print("We have now {0} sub solutions".format(len(rform.keys())))
return rform
def solution_var(self):
rform = self.reduce_exprs()
ks = list(rform.keys())
xs = [k for k in Puzzle.Symbols if k not in ks][0]
feasible_sols = [i for i in range(1, 10)]
[
feasible_sols.remove(x)
for x in rform.values()
if not isinstance(x, smp.add.Add)
]
soln = self.recurse_reduce(rform, xs, feasible_sols)
return soln
def recurse_reduce(self, exprs, xs, feasible_sols):
# print(feasible_sols)
rform = exprs
for S, F in rform.items():
for pvr in feasible_sols:
if isinstance(F, smp.numbers.Integer):
continue
if F.subs(xs, pvr) < 1 or F.subs(xs, pvr) > 9:
print(S, " = ", F.subs(xs, pvr),
f" when {str(xs)} = ", pvr)
feasible_sols.remove(pvr)
else:
continue
print(feasible_sols)
if len(feasible_sols) == 1:
print(xs, feasible_sols)
return xs, feasible_sols[0]
else:
return self.recurse_reduce(rform, xs, feasible_sols)
def solve(self):
sols = self.reduce_exprs()
vsub = self.solution_var()
for S, F in sols.items():
self.solution[str(S)] = F.subs(*vsub)
self.solution[str(vsub[0])] = vsub[1]
print(self.solution)
...