-
Notifications
You must be signed in to change notification settings - Fork 258
/
Copy pathword-ladder-ii.cpp
75 lines (67 loc) · 2.37 KB
/
word-ladder-ii.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
// Time: O(n * d), n is length of the string, d is size of the dictionary
// Space: O(d)
class Solution {
public:
/**
* @param start, a string
* @param end, a string
* @param dict, a set of string
* @return a list of lists of string
*/
vector<vector<string>> findLadders(string start, string end, unordered_set<string> &dict) {
unordered_map<string, unordered_set<string>> trace;
// BFS, each round is a level
unordered_set<string> levels[2];
int cur_level = 0;
levels[cur_level].emplace(start);
while (dict.size() > 0 && levels[cur_level % 2].size() > 0) {
// end is in current level, stop
if (trace.count(end) == 1) {
++cur_level;
break;
}
// clear words from the dictionary to avoid it' apear in future level again
for (auto& word : levels[cur_level % 2]) {
dict.erase(word);
}
levels[(cur_level + 1) % 2].clear();
for (auto& from : levels[cur_level % 2]) {
string new_word = from;
for (int i = 0; i < new_word.length(); ++i) {
char orig_c = from[i];
for (char c = 'a'; c <= 'z'; ++c) {
new_word[i] = c;
if (dict.count(new_word) == 1) {
trace[new_word].emplace(from);
levels[(cur_level + 1) % 2].emplace(new_word);
}
}
new_word[i] = orig_c;
}
}
++cur_level;
}
vector<vector<string>> result;
vector<string> temp;
traceBack(trace, temp, end, result);
return result;
}
void traceBack(
unordered_map<string, unordered_set<string>> &trace,
vector<string> &temp,
string end,
vector<vector<string>> &result) {
temp.emplace_back(end);
if (trace.count(end) != 1) {
vector<string> path = temp;
reverse(path.begin(), path.end());
result.emplace_back(move(path));
}
else {
for (auto& word: trace[end]) {
traceBack(trace, temp, word, result);
}
}
temp.pop_back();
}
};