-
Notifications
You must be signed in to change notification settings - Fork 28
/
Copy pathDesign_Search_Autocomplete_System.cpp
104 lines (97 loc) · 3.07 KB
/
Design_Search_Autocomplete_System.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
class AutocompleteSystem {
class TrieNode {
public:
bool mark;
int score;
unordered_map<char, TrieNode*> childrenMap;
TrieNode(): mark(false), score(0) {
childrenMap = unordered_map<char, TrieNode*>();
}
};
class Trie {
public:
TrieNode* root;
Trie(): root(new TrieNode()) {
}
void insert(string const& key, int score = 1) {
int n = (int)key.length();
TrieNode* pCrawl = root;
for(int i = 0; i < n; i++) {
if(!pCrawl->childrenMap[key[i]]) {
pCrawl->childrenMap[key[i]] = new TrieNode();
}
pCrawl = pCrawl->childrenMap[key[i]];
}
pCrawl->mark = true;
pCrawl->score += score;
}
void searchUtil(string& key, TrieNode* pCrawl, priority_queue<pair<int, string>>& pQ) {
if(!pCrawl) { return; }
if(pCrawl->mark) {
pair<int, string> curr = {-pCrawl->score, key};
if(pQ.size() < 3) {
pQ.push(curr);
}
else if(pQ.size() >= 3 and pQ.top() > curr) {
pQ.pop();
pQ.push(curr);
}
}
for(auto it = pCrawl->childrenMap.begin(); it != pCrawl->childrenMap.end(); ++it) {
key += it->first;
searchUtil(key, it->second, pQ);
key.pop_back();
}
}
vector<string> search(string& searchKey, TrieNode*& pCrawl) {
vector<string> result;
char key = searchKey[searchKey.length() - 1];
if(pCrawl->childrenMap[key]) {
pCrawl = pCrawl->childrenMap[key];
priority_queue<pair<int, string>> pQ;
searchUtil(searchKey, pCrawl, pQ);
result.resize(pQ.size());
int i = (int)result.size() - 1;
while(!pQ.empty()) {
result[i--] = pQ.top().second;
pQ.pop();
}
}
return result;
}
};
string queryKey;
Trie* trie;
TrieNode* curr;
bool found;
public:
AutocompleteSystem(vector<string> sentences, vector<int> times) {
queryKey = "";
trie = new Trie();
curr = trie->root;
found = true;
int n = (int)sentences.size();
assert(sentences.size() == times.size());
for(int i = 0; i < n; i++) {
trie->insert(sentences[i], times[i]);
}
}
vector<string> input(char c) {
if(c == '#') {
trie->insert(queryKey);
queryKey = "";
found = true;
curr = trie->root;
return vector<string>();
}
queryKey += c;
vector<string> result;
if(found) {
result = trie->search(queryKey, curr);
if(result.empty()) {
found = false;
}
}
return result;
}
};