-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathmrpt.go
217 lines (175 loc) · 4.49 KB
/
mrpt.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
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
package ann
import (
"math/rand"
"sort"
mat "github.com/gonum/matrix/mat64"
)
type mrpt struct {
xs [][]float64
trees []*tree
}
type tree struct {
root *node
r mat.Matrix
}
type node struct {
split float64
left *node
right *node
indices []int
}
// NewMRPTANNer creates a NN index using random projection trees
// See https://arxiv.org/pdf/1509.06957.pdf for additional details
// t -> number of trees, l -> depth of tree
func NewMRPTANNer(t int, l int, xs [][]float64) ANNer {
a := 0.5 // TODO(temporary)
// Infer dimensions from input matrix
n, d := len(xs), len(xs[0])
// Convert xs to a gonum matrix
X := mat.NewDense(d, n, nil)
for i := 0; i < n; i++ {
X.SetCol(i, xs[i])
}
return &mrpt{
xs: xs,
trees: growTrees(X, t, l, a),
}
}
func growTrees(X mat.Matrix, t, l int, a float64) []*tree {
// feature vector dimension
d, n := X.Dims()
trees := []*tree{}
// Create t trees
for i := 0; i < t; i++ {
// Create a RP matrix
R := mat.NewDense(d, l-1, nil)
// Create one random vector per tree level
for j := 0; j < (l - 1); j++ {
rs := []float64{}
for k := 0; k < d; k++ {
// TODO: only use non-zero value with prob a
// TODO: Use a sparse vector strategy
rs = append(rs, rand.NormFloat64())
}
// Set the random vector into the matrix
R.SetCol(j, rs)
}
// Calculate the projections
var P mat.Dense
P.Mul(X.T(), R)
// Create indices list
indices := []int{}
for i := 0; i < n; i++ {
indices = append(indices, i)
}
trees = append(trees, &tree{
r: R,
root: growTree(&P, l, 0, indices),
})
}
return trees
}
func growTree(P mat.Matrix, l, level int, indices []int) *node {
// Stop if we're at the leaf
if level == l-1 {
return &node{indices: indices}
}
// Get the projections for this level
ps := mat.Col(nil, level, P)
// Get the median of the projections
m := median(ps)
// Divide indices to left and right based on median value
leftIndices := []int{}
rightIndices := []int{}
for _, i := range indices {
if ps[i] <= m {
leftIndices = append(leftIndices, i)
} else {
rightIndices = append(rightIndices, i)
}
}
return &node{
left: growTree(P, l, level+1, leftIndices),
right: growTree(P, l, level+1, rightIndices),
split: m,
}
}
func (nn *mrpt) ANN(q []float64, k int) []int {
// Keep track of votes
votesMap := map[int]int{}
// How many votes does a vector need to be included in the output set
reqVotes := 1
// Convert q to a vector
qVec := mat.NewVector(len(q), q)
// Query the trees to get candidates
for _, tree := range nn.trees {
// Calculate projections for the query vector
var psVec mat.Vector
psVec.MulVec(tree.r.T(), qVec)
ps := mat.Col(nil, 0, &psVec)
indices := querySubtree(tree.root, ps, 0, k)
for _, i := range indices {
// Count vote
votesMap[i]++
}
}
xsCandidates := [][]float64{}
xsIndices := []int{}
for i, votes := range votesMap {
if votes >= reqVotes {
xsCandidates = append(xsCandidates, nn.xs[i])
// Track the index of each vector so we can retrieve it later
xsIndices = append(xsIndices, i)
}
}
// Perform naive k-nearest-neighbor search on candidates set
knn := NewExhaustiveNNer(xsCandidates)
knnIndices := knn.ANN(q, k)
// Convert the above knnIndices to the indices of the vectors in the scope
// of all of our data
indices := []int{}
for _, i := range knnIndices {
indices = append(indices, xsIndices[i])
}
return indices
}
func querySubtree(node *node, ps []float64, level int, k int) []int {
// Stop when we have reached the bottom
if level == len(ps) {
return node.indices
}
// Get the projection for the current level
p := ps[level]
indices := []int{}
if p <= node.split {
// Left branch
indices = querySubtree(node.left, ps, level+1, k)
// If not enough found on left, try right
if len(indices) < k {
indices = append(indices, querySubtree(node.right, ps, level+1, k)...)
}
} else {
// Right branch
indices = querySubtree(node.right, ps, level+1, k)
// If not enough found on right, try left
if len(indices) < k {
indices = append(indices, querySubtree(node.left, ps, level+1, k)...)
}
}
return indices
}
// median calculates the median value of a series of elements
func median(vals []float64) float64 {
// Make a copy so we don't alter the given slice
vs := make([]float64, len(vals))
copy(vs, vals)
sort.Float64s(vs)
// Even number of elements
if len(vs)%2 == 0 {
m := (vs[len(vs)/2-1] + vs[len(vs)/2]) / 2
return m
}
// Odd number of elements
m := vs[len(vs)/2]
return m
}