-
Notifications
You must be signed in to change notification settings - Fork 10
/
Copy pathtrie.go
148 lines (129 loc) · 3.21 KB
/
trie.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
// Package go-succinct-data-structure-trie implements trie with succinct data
// structure in Go.
package bits
import "unicode/utf8"
// https://blog.golang.org/strings
// https://golang.org/pkg/unicode/utf8/
/**
A Trie node, for use in building the encoding trie. This is not needed for
the decoder.
*/
type TrieNode struct {
letter string
final bool
children []*TrieNode
}
type Trie struct {
previousWord string
root *TrieNode
cache []*TrieNode
nodeCount uint
}
func (t *Trie) Init() {
t.previousWord = ""
t.root = &TrieNode{
letter: " ",
final: false,
}
t.cache = append(t.cache, t.root)
t.nodeCount = 1
}
/**
Returns the number of nodes in the trie
*/
func (t *Trie) GetNodeCount() uint {
return t.nodeCount
}
/**
Inserts a word into the trie. This function is fastest if the words are
inserted in alphabetical order.
*/
func (t *Trie) Insert(word string) {
commonPrefixWidth := 0
commonRuneCount := 0
minRuneCount := utf8.RuneCountInString(word)
if minRuneCount > utf8.RuneCountInString(t.previousWord) {
minRuneCount = utf8.RuneCountInString(t.previousWord)
}
for ; commonRuneCount < minRuneCount; commonRuneCount++ {
runeValue1, width1 := utf8.DecodeRuneInString(word[commonPrefixWidth:])
runeValue2, _ := utf8.DecodeRuneInString(t.previousWord[commonPrefixWidth:])
if runeValue1 != runeValue2 {
break
}
commonPrefixWidth += width1
}
t.cache = t.cache[:commonRuneCount+1]
node := t.cache[commonRuneCount]
for i, w := commonPrefixWidth, 0; i < len(word); i += w {
// fix the bug if words not inserted in alphabetical order
isLetterExist := false
runeValue, width := utf8.DecodeRuneInString(word[i:])
w = width
for _, cld := range node.children {
if cld.letter == string(runeValue) {
t.cache = append(t.cache, cld)
node = cld
isLetterExist = true
break
}
}
if isLetterExist {
continue
}
next := &TrieNode{
letter: string(runeValue),
final: false,
}
t.nodeCount++
node.children = append(node.children, next)
t.cache = append(t.cache, next)
node = next
}
node.final = true
t.previousWord = word
}
/**
Apply a function to each node, traversing the trie in level order.
*/
func (t *Trie) Apply(fn func(*TrieNode)) {
var level []*TrieNode
level = append(level, t.root)
for len(level) > 0 {
node := level[0]
level = level[1:]
for i := 0; i < len(node.children); i++ {
level = append(level, node.children[i])
}
fn(node)
}
}
/**
Encode the trie and all of its nodes. Returns a string representing the
encoded data.
*/
func (t *Trie) Encode() string {
// Write the unary encoding of the tree in level order.
bits := BitWriter{}
bits.Write(0x02, 2)
t.Apply(func(node *TrieNode) {
for i := 0; i < len(node.children); i++ {
bits.Write(1, 1)
}
bits.Write(0, 1)
})
// Write the data for each node, using (dataBits) bits for one node.
// 1 bit stores the "final" indicator. The other (dataBits-1) bits store
// one of the characters of the alphabet.
t.Apply(func(node *TrieNode) {
value, ok := mapCharToUint[node.letter]
if !ok {
panic("illegal character:" + node.letter)
}
if node.final {
value |= (1 << (dataBits - 1))
}
bits.Write(uint(value), dataBits)
})
return bits.GetData()
}