-
-
Notifications
You must be signed in to change notification settings - Fork 1.1k
/
Copy pathHeapSort.java
84 lines (69 loc) · 1.94 KB
/
HeapSort.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
78
79
80
81
82
83
84
import java.util.Arrays;
/**
* 堆排序
*/
public class HeapSort implements IArraySort {
@Override
public int[] sort(int[] sourceArray) throws Exception {
// 对 arr 进行拷贝,不改变参数内容
int[] arr = Arrays.copyOf(sourceArray, sourceArray.length);
int len = arr.length;
buildMaxHeap(arr, len);
for (int i = len - 1; i > 0; i--) {
swap(arr, 0, i);
len--;
heapify(arr, 0, len);
}
return arr;
}
private void buildMaxHeap(int[] arr, int len) {
for (int i = (int) Math.floor(len / 2); i >= 0; i--) {
heapify(arr, i, len);
}
}
/**
* 递归方式堆化
*/
private void heapifyRecursive(int[] arr, int i, int len) {
int left = 2 * i + 1;
int right = 2 * i + 2;
int largest = i;
if (left < len && arr[left] > arr[largest]) {
largest = left;
}
if (right < len && arr[right] > arr[largest]) {
largest = right;
}
if (largest != i) {
swap(arr, i, largest);
heapifyRecursive(arr, largest, len);
}
}
/**
* 非递归方式堆化
*/
private void heapify(int[] arr, int i, int len) {
int tmp = arr[i];
for (int j = i * 2 + 1; j < len; j = j * 2 + 1) {
//比较左右子结点谁更大
if (j + 1 < len && arr[j] < arr[j + 1]) {
j++;
}
//如果子节点没父节点大,跳过
if (arr[j] <= tmp) {
break;
} else {
//将子节点的值赋给父节点
arr[i] = arr[j];
//升级为新的父节点
i = j;
}
}
arr[i] = tmp;
}
private void swap(int[] arr, int i, int j) {
int temp = arr[i];
arr[i] = arr[j];
arr[j] = temp;
}
}