-
-
Notifications
You must be signed in to change notification settings - Fork 16
/
Copy pathsudoku_test.go
118 lines (109 loc) · 2.38 KB
/
sudoku_test.go
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
package gini_test
import (
"fmt"
"github.com/go-air/gini"
"github.com/go-air/gini/z"
)
func Example_sudoku() {
g := gini.New()
// 9 rows, 9 cols, 9 boxes, 9 numbers
// one variable for each triple (row, col, n)
// indicating whether or not the number n
// appears in position (row,col).
var lit = func(row, col, num int) z.Lit {
n := num
n += col * 9
n += row * 81
return z.Var(n + 1).Pos()
}
// add a clause stating that every position on the board has a number
for row := 0; row < 9; row++ {
for col := 0; col < 9; col++ {
for n := 0; n < 9; n++ {
m := lit(row, col, n)
g.Add(m)
}
g.Add(0)
}
}
// every row has unique numbers
for n := 0; n < 9; n++ {
for row := 0; row < 9; row++ {
for colA := 0; colA < 9; colA++ {
a := lit(row, colA, n)
for colB := colA + 1; colB < 9; colB++ {
b := lit(row, colB, n)
g.Add(a.Not())
g.Add(b.Not())
g.Add(0)
}
}
}
}
// every column has unique numbers
for n := 0; n < 9; n++ {
for col := 0; col < 9; col++ {
for rowA := 0; rowA < 9; rowA++ {
a := lit(rowA, col, n)
for rowB := rowA + 1; rowB < 9; rowB++ {
b := lit(rowB, col, n)
g.Add(a.Not())
g.Add(b.Not())
g.Add(0)
}
}
}
}
// function adding constraints stating that every box on the board
// rooted at x, y has unique numbers
var box = func(x, y int) {
// all offsets w.r.t. root x,y
offs := []struct{ x, y int }{{0, 0}, {0, 1}, {0, 2}, {1, 0}, {1, 1}, {1, 2}, {2, 0}, {2, 1}, {2, 2}}
// all numbers
for n := 0; n < 9; n++ {
for i, offA := range offs {
a := lit(x+offA.x, y+offA.y, n)
for j := i + 1; j < len(offs); j++ {
offB := offs[j]
b := lit(x+offB.x, y+offB.y, n)
g.Add(a.Not())
g.Add(b.Not())
g.Add(0)
}
}
}
}
// every box has unique numbers
for x := 0; x < 9; x += 3 {
for y := 0; y < 9; y += 3 {
box(x, y)
}
}
if g.Solve() != 1 {
fmt.Printf("error, unsat sudoku.\n")
return
}
for row := 0; row < 9; row++ {
for col := 0; col < 9; col++ {
for n := 0; n < 9; n++ {
if g.Value(lit(row, col, n)) {
fmt.Printf("%d", n+1)
break
}
}
if col != 8 {
fmt.Printf(" ")
}
}
fmt.Printf("\n")
}
// Output: 5 2 9 1 3 6 7 4 8
// 4 3 1 7 8 5 2 9 6
// 8 7 6 4 9 2 1 3 5
// 1 6 3 2 4 8 5 7 9
// 2 4 5 9 1 7 8 6 3
// 7 9 8 5 6 3 4 1 2
// 6 5 4 3 2 1 9 8 7
// 3 1 2 8 7 9 6 5 4
// 9 8 7 6 5 4 3 2 1
}