-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathwalking_distance.go
180 lines (163 loc) · 4.15 KB
/
walking_distance.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
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
// implementation of walking distance heurestic
package main
import (
"math/bits"
)
var bitsLen = bits.Len(uint(BOARD_ROW_SIZE))
type visitSt struct {
cost t_cell
board [16]t_cell
e t_cell
}
// creates a new visit struct
func nVisit(cost t_cell, board [16]t_cell, e t_cell) *visitSt {
return &visitSt{
cost,
board,
e,
}
}
// Generates a unique int representation of the board. Generates the same int for the same board every time, becuase it shifts the bits according to the value and index positions
func code(board [16]t_cell) int {
r := 0
for i := range board {
r |= int(board[i]) << (bitsLen * i)
}
return r
}
func codeUniq(board [16]t_cell) int {
r := 0
for i := range board {
r |= int(board[i]) << (bitsLen*i + i)
}
return r
}
type WalkingDistance struct {
// walking distance table
table map[int]t_cell
// key is the value of the correct cell
// "map value int" is the index to the correct value cell
solved map[int]int
// BOARD_ROW_SIZE
size int
// BOARD_ROW_SIZE bit length
bitLength int
}
func NewWD(rowSize int) *WalkingDistance {
wd := &WalkingDistance{
size: rowSize,
bitLength: bitsLen,
}
solvedArr := startingPoint(t_cell(wd.size))
// map representation of solved values to used for faster lookup time
wd.solved = make(map[int]int)
for i := 0; i < 16; i++ {
wd.solved[int(solvedArr[i])] = i
}
wd.GenerateTable()
return wd
}
func (wd *WalkingDistance) GenerateTable() *WalkingDistance {
wd.table = make(map[int]t_cell)
size := t_cell(wd.size)
solved := [16]t_cell{4, 0, 0, 0, 0, 4, 0, 0, 0, 0, 4, 0, 0, 0, 0, 3}
visitable := make(chan *visitSt, 92850)
visitable <- nVisit(0, solved, size-1)
count := 0
// precalculate the walking distance table
for visit := range visitable {
key := code(visit.board)
if _, found := wd.table[key]; found {
continue
}
wd.table[key] = visit.cost
// calculates the horizontal or vertical heuristics of a board
for _, d := range []t_cell{-1, 1} {
if 0 <= (visit.e+d) && (visit.e+d) < size {
var i t_cell
for i < size {
if visit.board[size*(visit.e+d)+i] > 0 {
var newBoard [16]t_cell
copy(newBoard[:], visit.board[:])
newBoard[size*(visit.e+d)+i] -= 1
newBoard[size*visit.e+i] += 1
visitable <- nVisit(visit.cost+1, newBoard, visit.e+d)
count++
}
i++
}
}
}
// check if the maximum amount of table items has been created
if count == 92850 {
close(visitable)
}
}
return wd
}
// calculate the walking distance from the given board
func (wd *WalkingDistance) Calculate(board [16]t_cell) int {
heurestic := 0
rowCode := 0
colCode := 0
for i, val := range board {
if val == 0 {
continue
}
// index that the value should be set to
corIdx := wd.solved[int(val)]
// vertical and horizontal indexs of the current position
xi, yi := i%wd.size, i/wd.size
// xCor = vertical index of the correct position
// yCor = horizontal index of the correct position
xCor, yCor := corIdx%wd.size, corIdx/wd.size
// craete the vertical (row) and horizontal (col) codes that point to the walking distance table
rowCode += 1 << (wd.bitLength * (wd.size*yi + yCor))
colCode += 1 << (wd.bitLength * (wd.size*xi + xCor))
if yCor == yi {
// calculate vertical heursitic increments
for yInc := i + 1; yInc < i-i%wd.size+wd.size; yInc++ {
if int(yInc) < len(board) {
yVal := wd.solved[int(board[int(yInc)])]
if yVal/wd.size == yi && yVal < corIdx {
heurestic += 2
}
}
}
}
if xCor == xi {
// calculate horizontal heursitic increments
for xInc := i + wd.size; xInc < wd.size*wd.size; xInc += 4 {
if xInc < len(board) {
kVal := wd.solved[int(board[int(xInc)])]
if kVal%wd.size == xi && kVal < corIdx {
heurestic += 2
}
}
}
}
}
// sum heuristic values together.
heurestic += int(wd.table[rowCode] + wd.table[colCode])
return heurestic
}
// util functions for debugging mostly
func sum(l []t_cell) int {
sum := 0
for _, v := range l {
sum += int(v)
}
return sum
}
func avg(l []t_cell) int {
return sum(l) / len(l)
}
func max(l []t_cell) t_cell {
var max t_cell
for _, v := range l {
if v > max {
max = v
}
}
return max
}