-
Notifications
You must be signed in to change notification settings - Fork 100
/
Copy pathinversions-count.ts
60 lines (51 loc) · 1.74 KB
/
inversions-count.ts
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
/**
* Given array A[1... n], for every i < j, find all inversion pairs such that A[i] > A[j].
* Solution must have complexity O(n*lg(n)).
*
* Tip: modify merge-sort
*/
import { range } from '../../utils';
import head from 'lodash/head';
/**
* Divide and sort merges two subarrays of given array, like in merge-sort, but
* count also the number of inversions in the merged array during the process.
* Time complexity: O(n)
* @param input The array which subarrays should be sorted
* @param start Start of the first array
* @param mid End of first array, not included
* @param end End of second array, not including
*/
function merge(input: number[], start: number, mid: number, end: number): number {
const left = input.slice(start, mid);
const right = input.slice(mid, end);
left[left.length] = -Infinity as any;
right[right.length] = -Infinity as any;
let inversionCount = 0;
range(start, end - 1).forEach(index => {
if (head(left) > head(right)) {
inversionCount += right.length - 1; // Do not include Infinity in the count
input[index] = left.shift();
} else {
input[index] = right.shift();
}
});
return inversionCount;
}
/**
* Count the number of inversions
* Time complexity: O(n*lg(n)).
* @param input The array which should be sorted
* @param start Left side of the subarray
* @param end Right side of the subarray, not included
*/
export function countInversions(
input: number[],
start: number = 0,
end: number = input.length
): number {
if (end - start <= 1) return 0;
const mid = Math.floor((start + end) / 2);
const leftCount = countInversions(input, start, mid);
const rightCount = countInversions(input, mid, end);
return leftCount + rightCount + merge(input, start, mid, end);
}