-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathalien_dictionary.rs
204 lines (177 loc) · 5.89 KB
/
alien_dictionary.rs
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
use std::collections::HashMap;
use std::collections::HashSet;
use std::collections::VecDeque;
enum ComparisonResult {
ComesBefore(char, char),
Impossible,
NoInformation,
}
/// There is a new alien language that uses the English alphabet. However, the
/// order among the letters is unknown to you.
///
/// You are given a list of strings `words` from the alien language's
/// dictionary, where the strings in `words` are sorted lexicographically by
/// the rules of this new language.
///
/// Return a string of the unique letters in the new alien language sorted in
/// lexicographically increasing order by the new language's rules. If there is
/// no solution, return `""`. If there are multiple solutions, return any of
/// them.
struct Solution;
impl Solution {
fn compare_words(before: &str, after: &str) -> ComparisonResult {
let mut bchars = before.chars();
let mut achars = after.chars();
let result;
loop {
match (bchars.next(), achars.next()) {
(Some(b), Some(a)) => {
if b != a {
result = ComparisonResult::ComesBefore(b, a);
break;
}
}
(Some(_), None) => {
result = ComparisonResult::Impossible;
break;
}
_ => {
result = ComparisonResult::NoInformation;
break;
}
}
}
result
}
pub fn alien_order(words: Vec<String>) -> String {
let n = words.len();
let mut in_degrees = HashMap::new();
let mut comes_before = HashMap::new();
// Initiale in_degrees
for i in 0..n {
let word = &words[i];
for c in word.chars() {
in_degrees
.entry(c)
.or_insert(0);
}
}
for i in 0..n-1 {
let before = &words[i];
let after = &words[i+1];
let comparison = Self::compare_words(before, after);
match comparison {
ComparisonResult::ComesBefore(b, a) => {
let inserted = comes_before
.entry(b)
.or_insert(HashSet::new())
.insert(a);
if inserted {
in_degrees
.entry(a)
.and_modify(|count| { *count += 1; })
.or_insert(1);
}
}
ComparisonResult::NoInformation => { } // this is ok
ComparisonResult::Impossible => {
return "".to_string()
}
}
}
let mut seen = HashSet::new();
let mut queue = VecDeque::new();
let mut result = Vec::new();
for (k, v) in in_degrees.iter() {
if *v == 0 {
queue.push_back(*k);
seen.insert(*k);
result.push(*k);
}
}
while !queue.is_empty() {
let letter = queue.pop_front().unwrap();
if comes_before.contains_key(&letter) {
for other in &comes_before[&letter] {
if !seen.contains(other) {
in_degrees
.entry(*other)
.and_modify(|count| { *count -= 1; });
if in_degrees[other] == 0 {
queue.push_back(*other);
seen.insert(*other);
result.push(*other);
}
}
}
}
}
if seen.len() == in_degrees.len() {
result.iter().collect::<String>()
} else {
"".to_string()
}
}
}
#[cfg(test)]
mod tests {
use super::Solution;
// t -> f
// w -> e
// r -> t
// e -> r
// wertf
#[test]
fn example_1() {
let orig = vec!["wrt", "wrf", "er", "ett", "rftt"];
let words = orig.iter().map(|s| s.to_string()).collect();
let result = Solution::alien_order(words);
assert_eq!(result, "wertf");
}
#[test]
fn example_2() {
let orig = vec!["z", "x"];
let words = orig.iter().map(|s| s.to_string()).collect();
let result = Solution::alien_order(words);
assert_eq!(result, "zx");
}
#[test]
fn example_3() {
let orig = vec!["z", "x", "z"];
let words = orig.iter().map(|s| s.to_string()).collect();
let result = Solution::alien_order(words);
assert_eq!(result, "");
}
#[test]
fn example_longer() {
let orig = vec!["z", "zx"];
let words = orig.iter().map(|s| s.to_string()).collect();
let result = Solution::alien_order(words);
// Return any of them. Not sure the ordering between z and x
assert!(result == "xz" || result == "zx");
}
#[test]
fn same() {
let orig = vec!["z", "z"];
let words = orig.iter().map(|s| s.to_string()).collect();
let result = Solution::alien_order(words);
assert_eq!(result, "z");
}
#[test]
fn unknown() {
let orig = vec!["zy", "zx"];
let words = orig.iter().map(|s| s.to_string()).collect();
let result = Solution::alien_order(words);
// Return any of them. y comes before x, but don't know z
let options = vec!["yzx", "yxz", "zyx"];
let options: Vec<String> = options.iter().map(|s| s.to_string()).collect();
assert!(options.contains(&result));
}
#[test]
fn impossible() {
let orig = vec!["abc", "ab"];
let words = orig.iter().map(|s| s.to_string()).collect();
let result = Solution::alien_order(words);
assert_eq!(result, "");
}
}