forked from alexfertel/rust-algorithms
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathknuth_morris_pratt.rs
118 lines (94 loc) · 2.48 KB
/
knuth_morris_pratt.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
fn precompute_table(pattern: &str) -> Vec<usize> {
let p = pattern.as_bytes();
let mut pi = vec![0; pattern.len()];
let mut k = 0;
for q in 1..p.len() {
while k > 0 && p[k] != p[q] {
k = pi[k];
}
if p[k] == p[q] {
k += 1;
}
pi[q] = k;
}
pi
}
pub fn knuth_morris_pratt(text: &str, pattern: &str) -> Vec<usize> {
if text.is_empty() || pattern.is_empty() {
return vec![];
}
let t = text.as_bytes();
let p = pattern.as_bytes();
let pi = precompute_table(pattern);
let mut matches = vec![];
let mut q = 0;
for i in 1..=t.len() {
while q > 0 && p[q] != t[i - 1] {
q = pi[q - 1];
}
if p[q] == t[i - 1] {
q += 1;
}
if q == p.len() {
matches.push(i - p.len());
q = pi[q - 1];
}
}
matches
}
#[cfg(test)]
mod test {
use super::{knuth_morris_pratt, precompute_table};
#[test]
fn builds_pi_correctly() {
let pi = precompute_table("ababaca");
assert_eq!(pi, vec![0, 0, 1, 2, 3, 0, 1]);
}
#[test]
fn each_letter_matches() {
let pi = precompute_table("aaa");
assert_eq!(pi, vec![0, 1, 2]);
let index = knuth_morris_pratt("aaa", "a");
assert_eq!(index, vec![0, 1, 2]);
}
#[test]
fn a_few_separate_matches() {
let index = knuth_morris_pratt("abababa", "ab");
assert_eq!(index, vec![0, 2, 4]);
}
#[test]
fn one_match() {
let index = knuth_morris_pratt("ABC ABCDAB ABCDABCDABDE", "ABCDABD");
assert_eq!(index, vec![15]);
}
#[test]
fn lots_of_matches() {
let index = knuth_morris_pratt("aaabaabaaaaa", "aa");
assert_eq!(index, vec![0, 1, 4, 7, 8, 9, 10]);
}
#[test]
fn lots_of_intricate_matches() {
let index = knuth_morris_pratt("ababababa", "aba");
assert_eq!(index, vec![0, 2, 4, 6]);
}
#[test]
fn not_found0() {
let index = knuth_morris_pratt("abcde", "f");
assert_eq!(index, vec![]);
}
#[test]
fn not_found1() {
let index = knuth_morris_pratt("abcde", "ac");
assert_eq!(index, vec![]);
}
#[test]
fn not_found2() {
let index = knuth_morris_pratt("ababab", "bababa");
assert_eq!(index, vec![]);
}
#[test]
fn empty_string() {
let index = knuth_morris_pratt("", "abcdef");
assert_eq!(index, vec![]);
}
}