-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathTrie.cpp
141 lines (115 loc) · 3.51 KB
/
Trie.cpp
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
#include <vector>
#include <unordered_map>
#include <string>
#include <algorithm>
#include <iostream>
using namespace std;
class Trie {
public:
Trie() : root(new node('0')) {}
~Trie() {
delete root;
}
// insert a word to the tree, and add index to the node of the word
// O(M), M is the length of input string
void insert(string & input, int index) {
int count = 0;
node * trav = root;
for (auto s : input) {
auto found = trav->child.find(s);
if (found != trav->child.end()) {
trav = found->second;
}
else {
trav->child[s] = new node(s);
found = trav->child.find(s);
trav = found->second;
}
count++;
if (count == input.size()) {
trav->wordIndex.push_back(index);
}
}
}
// find words that has the given prefixes and store index to result
// search prefixes take O(M), find result takes O(MN) in the worest case
void findprefixes(string & input, vector<int> & result) {
int count = 0;
node * trav = root;
for (auto s : input) {
auto found = trav->child.find(s);
if (found != trav->child.end()) {
trav = found->second;
}
else {
return; // empty
}
count++;
if (count == input.size()) {
findchild(trav, result);
return;
}
}
}
private:
// node in the tree
class node {
public:
node(char chr) : chr(chr) {}
node() : chr('0') {}
~node() {
for (auto & i : child) {
delete i.second;
}
}
char chr;
unordered_map<char, node *> child;
vector<int> wordIndex;
};
node * root;
// preorder traverse
void findchild(node * trav, vector<int> & result) {
for (auto index : trav->wordIndex) {
result.push_back(index);
}
if (trav->child.size() == 0) {
return;
}
for (auto & n : trav->child) {
findchild(n.second, result);
}
}
};
// replace words in the sentence to its prefixes
vector<string> replace_words(const vector<string> & prefixes, const vector<string> & sentence) {
Trie dictionary_tree;
vector<string> new_sentence = sentence;
// build trie, O(NM), N is then number of words, M is average length
for (int i = 0; i < new_sentence.size(); i++) {
dictionary_tree.insert(new_sentence[i], i);
}
vector<string> sorted_prefixes = prefixes;
sort(sorted_prefixes.begin(), sorted_prefixes.end(),
[](const string & a, const string & b) -> bool
{
return a.size() > b.size();
}); // sort by size of prefixes in desending order,
// so will finally replace by shortest prefixes in next step
// search and replace words
// O(PNM) in worest case, O(PM) in best case
for (int i = 0; i < sorted_prefixes.size(); i++) {
vector<int> result;
dictionary_tree.findprefixes(sorted_prefixes[i], result);
for (int index : result) {
new_sentence[index] = sorted_prefixes[i];
}
}
return new_sentence;
}
int main() { //test
vector<string> temp;
temp = replace_words({ "cat", "bat", "rat" }, { "the", "cattle", "was", "rattled", "by", "the", "battery" });
for (auto s : temp) {
cout << s << endl;
}
}