forked from mirrors/jj
diff: fix LCS when a line/word/byte has been moved later
This commit is contained in:
parent
13134bd5a4
commit
5b18e89a4d
1 changed files with 24 additions and 9 deletions
|
@ -160,26 +160,33 @@ fn find_lcs(input: &[usize]) -> Vec<(usize, usize)> {
|
||||||
}
|
}
|
||||||
|
|
||||||
let mut chain = vec![(0, 0, 0); input.len()];
|
let mut chain = vec![(0, 0, 0); input.len()];
|
||||||
let mut longest = 0;
|
let mut global_longest = 0;
|
||||||
let mut longest_right_pos = 0;
|
let mut global_longest_right_pos = 0;
|
||||||
for (right_pos, &left_pos) in input.iter().enumerate() {
|
for (right_pos, &left_pos) in input.iter().enumerate() {
|
||||||
chain[right_pos] = (1, left_pos, usize::MAX);
|
let mut longest_from_here = 1;
|
||||||
|
let mut previous_right_pos = usize::MAX;
|
||||||
for i in (0..right_pos).rev() {
|
for i in (0..right_pos).rev() {
|
||||||
let (previous_len, previous_left_pos, _) = chain[i];
|
let (previous_len, previous_left_pos, _) = chain[i];
|
||||||
if previous_left_pos < left_pos {
|
if previous_left_pos < left_pos {
|
||||||
let len = previous_len + 1;
|
let len = previous_len + 1;
|
||||||
chain[right_pos] = (len, left_pos, i);
|
if len > longest_from_here {
|
||||||
if len > longest {
|
longest_from_here = len;
|
||||||
longest = len;
|
previous_right_pos = i;
|
||||||
longest_right_pos = right_pos;
|
if len > global_longest {
|
||||||
}
|
global_longest = len;
|
||||||
|
global_longest_right_pos = right_pos;
|
||||||
|
// If this is the longest chain globally so far, we cannot find a
|
||||||
|
// longer one by using a previous value, so break early.
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
chain[right_pos] = (longest_from_here, left_pos, previous_right_pos);
|
||||||
|
}
|
||||||
|
|
||||||
let mut result = vec![];
|
let mut result = vec![];
|
||||||
let mut right_pos = longest_right_pos;
|
let mut right_pos = global_longest_right_pos;
|
||||||
loop {
|
loop {
|
||||||
let (_, left_pos, previous_right_pos) = chain[right_pos];
|
let (_, left_pos, previous_right_pos) = chain[right_pos];
|
||||||
result.push((left_pos, right_pos));
|
result.push((left_pos, right_pos));
|
||||||
|
@ -578,6 +585,14 @@ mod tests {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_find_lcs_element_moved_later() {
|
||||||
|
assert_eq!(
|
||||||
|
find_lcs(&[0, 1, 3, 4, 2, 5, 6]),
|
||||||
|
vec![(0, 0), (1, 1), (3, 2), (4, 3), (5, 5), (6, 6)]
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn test_find_lcs_interleaved_longest_chains() {
|
fn test_find_lcs_interleaved_longest_chains() {
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
|
|
Loading…
Reference in a new issue