-
Notifications
You must be signed in to change notification settings - Fork 1.5k
/
Copy pathSolution.java
77 lines (67 loc) · 2.85 KB
/
Solution.java
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
// github.com/RodneyShag
import java.util.Scanner;
import java.util.Arrays;
// We basically implement MergeSort and
// 1) Add "swaps" counter and 1 line of code to count swaps when merging
// 2) Use "long" instead of "int" to avoid integer overflow
// Time Complexity: O(n log n)
// Space Complexity: O(n)
public class Solution {
public static void main(String[] args) {
Scanner scan = new Scanner(System.in);
int testcases = scan.nextInt();
while (testcases-- > 0) {
int n = scan.nextInt();
int [] array = new int[n];
for (int i = 0; i < n; i++) {
array[i] = scan.nextInt();
}
MergeSort ms = new MergeSort();
System.out.println(ms.mergeSort(array));
}
scan.close();
}
private static class MergeSort {
/* Our array has up to n = 100,000 elements. That means there may be O(n^2) swaps.
n^2 is 10,000,000,000. A Java int has max value 2,147,483,647 so we use a long
to avoid integer overflow */
private long swaps = 0;
public long mergeSort(int [] array) {
int [] helper = new int[array.length];
mergeSort(array, helper, 0, array.length - 1);
return swaps;
}
private void mergeSort(int [] array, int [] helper, int start, int end) {
if (start < end) {
int mid = (start + end) / 2;
mergeSort(array, helper, start, mid);
mergeSort(array, helper, mid + 1, end);
merge(array, helper, start, mid, end);
}
}
private void merge(int [] array, int [] helper, int start, int mid, int end) {
/* Fill helper array with same elements as original array */
for (int i = start; i <= end; i++) { // notice "i" goes from "start" to "end", not "0" to "array.length"
helper[i] = array[i];
}
int curr = start;
int left = start;
int right = mid + 1;
/* Loop through helper[] left and right halves and continuously copy smaller element to array[] */
while (left <= mid && right <= end) {
if (helper[left] <= helper[right]) {
array[curr++] = helper[left++];
} else {
/* Each time we choose element from right side, we count up how many elements
it is less than from left side. This is equivalent to counting swaps. */
swaps += mid + 1 - left;
array[curr++] = helper[right++];
}
}
/* Copy remaining elements of left half. Right half elements are already in proper place */
while (left <= mid) {
array[curr++] = helper[left++];
}
}
}
}