常用排序算法
冒泡排序 选择排序 快速排序 归并排序 堆排序 插入排序
计数排序 桶排序 希尔排序
名称 | 时间复杂度 | 空间复杂度 | 稳定性 | 备注 |
---|---|---|---|---|
冒泡 | O(n²) | O(1) | 稳定 | |
选择 | O(n²) | O(1) | 不稳定 | 5 8 5 2 9 最小元素交换的时候 |
快速排序 | O(nlogn) | O(logn) 递归 | 不稳定 | 5 3 3 4 3 8 9 10 11 ,基准元素交换的时候 |
归并排序 | O(nlogn) | O(n) | 稳定 | |
堆排序 | O(nlogn) | O(1) 堆是完全二叉树,直接可以用数组存储 | 不稳定 | 尾元素移动到顶部的时候 |
冒泡
选择
快排
归并
堆
快速排序
- 非递归-栈
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
56import java.util.*;
public class QuickSortNoRecursion {
private int partition(int[] array, int startIndex, int endIndex) {
int pivot = startIndex;
int mark = startIndex;
for (int i=startIndex+1; i<=endIndex; i++) {
if (array[i] < array[pivot]) {
mark++;
int temp = array[mark];
array[mark] = array[i];
array[i] = temp;
}
}
int temp = array[pivot];
array[pivot] = array[mark];
array[mark] = temp;
return mark;
}
private void quickSort(int[] array, int startIndex, int endIndex) {
Deque<Map<String, Integer>> deque = new LinkedList<>();
Map<String, Integer> map = new HashMap<>();
map.put("startIndex", startIndex);
map.put("endIndex", endIndex);
deque.addFirst(map);
while (!deque.isEmpty()) {
// 取参数
Map<String, Integer> parameter = deque.removeFirst();
int paramStart, paramEnd, paramPivot;
paramStart = parameter.get("startIndex");
paramEnd = parameter.get("endIndex");
// 分割
paramPivot = this.partition(array, paramStart, paramEnd);
// 基准左边开始结束参数入栈
if (paramStart < paramPivot-1) {
Map<String, Integer> leftParam = new HashMap<>();
leftParam.put("startIndex", paramStart);
leftParam.put("endIndex", paramPivot-1);
deque.addFirst(leftParam);
}
// 基准右边开始结束参数入栈
if (paramPivot+1 < paramEnd) {
Map<String, Integer> rightParam = new HashMap<>();
rightParam.put("startIndex", paramPivot+1);
rightParam.put("endIndex", paramEnd);
deque.addFirst(rightParam);
}
}
}
public static void main(String[] args) {
QuickSortNoRecursion quickSortNoRecursion = new QuickSortNoRecursion();
int[] array = {5, 4, 3, 8, 7, 6, 2, 1};
quickSortNoRecursion.quickSort(array, 0, array.length-1);
System.out.println(Arrays.toString(array));
}
} - 非递归-队列
发现栈与队列效果都一样,因为通过partition,数组已经被分割成了互不干扰的小块,小块的处理顺序不影响最终的排序结果,上述利用栈,每次都处理右侧的小块,而利用队列,逐级处理,先处理本层的左右小块,在处理下层更小的小块
1 | import java.util.*; |
- 递归
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
35import java.util.*;
public class QuickSortRecursion {
private int partition(int[] array, int startIndex, int endIndex) {
int pivot = startIndex;
int mark = startIndex;
for (int i=startIndex+1; i<=endIndex; i++) {
if (array[i] < array[pivot]) {
mark++;
this.swap(array, mark, i);
}
}
this.swap(array, pivot, mark);
return mark;
}
private void swap(int[] array, int src, int des) {
int temp = array[src];
array[src] = array[des];
array[des] = temp;
}
private void quickSort(int[] array, int startIndex, int endIndex) {
if (startIndex >= endIndex) {
return;
}
int pivotIndex = this.partition(array, startIndex, endIndex);
quickSort(array, startIndex, pivotIndex-1);
quickSort(array, pivotIndex+1, endIndex);
}
public static void main(String[] args) {
QuickSortRecursion quickSortRecursion = new QuickSortRecursion();
int[] array = {5, 4, 3, 8, 7, 6, 2, 1};
quickSortRecursion.quickSort(array, 0, array.length-1);
System.out.println(Arrays.toString(array));
}
}
选择排序
1 | public class SelectSort { |
归并排序
- 原数组排序
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
43import java.util.Arrays;
public class MergeSort {
public static void main(String[] args) {
MergeSort mergeSort = new MergeSort();
int[] array = {1, 2, 3, 6, 5, 4};
mergeSort.sort(array, 0, array.length-1);
System.out.println(Arrays.toString(array));
}
// sort函数使指定闭区间内的数组变成有序的, sort先将其切割成小区间,
// 实现排序功能的是靠merge函数
private void sort(int[] array, int startIndex, int endIndex) {
if (startIndex >= endIndex) {
return;
}
int middleIndex = startIndex + (endIndex - startIndex) / 2;
sort(array, startIndex, middleIndex);
sort(array, middleIndex+1, endIndex);
merge(array, startIndex, middleIndex, endIndex);
}
private void merge(int[] array, int startIndex, int middleIndex, int endIndex) {
int[] temp = new int[endIndex-startIndex+1];
int leftIndex = startIndex;
int rightIndex = middleIndex+1;
int mergeIndex = 0;
while (leftIndex<=middleIndex && rightIndex<=endIndex) {
if (array[leftIndex] < array[rightIndex]) {
temp[mergeIndex++] = array[leftIndex++];
} else {
temp[mergeIndex++] = array[rightIndex++];
}
}
while (leftIndex <= middleIndex) {
temp[mergeIndex++] = array[leftIndex++];
}
while (rightIndex <= endIndex) {
temp[mergeIndex++] = array[rightIndex++];
}
for (int i=0; i<temp.length; i++) {
array[i+startIndex] = temp[i];
}
}
} - 返回新数组
空间复杂度 递归的深度是 logn 在合并的时候需要创建数组存放合并后的数组 两个数组在合并后,没用引用指向它们,内存被释放,而合并在一起的新数组占用内存,在最后一次合并的时候占用的空间最大O(n)
时间复杂度 O(nlogn) 一共需要合并logn次,每一层,所有成对的小块合并,需要n次比较
1 | import java.util.Arrays; |
冒泡
1 | public ArrayList<Integer> GetLeastNumbers_Solution(int [] input, int k) { |
刷题
数组中重复的数字
- 利用集合
- 重排数组
从头开始遍历数组,如果当前数组的值和索引值不一致,把当前值交换到和索引值一致的位置,然后再判断当前值和索引值是否一致,不一致继续交换,一个萝卜一个坑,萝卜移动到相应的坑中,当发现坑中有相同的萝卜的时候,这个萝卜就是重复的。
1 | public int duplicate (int[] numbers) { |
数组中的逆序对
- 暴力法
1
2
3
4
5
6
7
8
9
10
11
12public int InversePairs(int [] array) {
int mod = 1000000007;
int result = 0;
for (int i=0; i<array.length; i++) {
for (int j=i+1; j<array.length; j++) {
if (array[i] > array[j]) {
result += 1;
}
}
}
return result % mod;
}
最小k个数
- 冒泡
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17public ArrayList<Integer> GetLeastNumbers_Solution(int [] input, int k) {
int length = input.length;
for (int i=0; i<length-1; i++) {
for (int j=0; j<length-1-i; j++) {
if (input[j] > input[j+1]) {
int temp = input[j];
input[j] = input[j+1];
input[j+1] = temp;
}
}
}
ArrayList<Integer> result = new ArrayList<>();
for (int i=0; i<k; i++) {
result.add(input[i]);
}
return result;
} - 选择
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21public ArrayList<Integer> GetLeastNumbers_Solution(int [] input, int k) {
int length = input.length;
for (int i=0; i<k; i++) {
int minIndex = i;
for (int j=i+1; j<length; j++) {
if (input[j] < input[minIndex]) {
minIndex = j;
}
}
if (minIndex != i) {
int temp = input[i];
input[i] = input[minIndex];
input[minIndex] = temp;
}
}
ArrayList<Integer> result = new ArrayList<>();
for (int i=0; i<k; i++) {
result.add(input[i]);
}
return result;
} - 快排
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
41public ArrayList<Integer> GetLeastNumbers_Solution(int [] input, int k) {
if (k == 0) {
return new ArrayList<Integer>();
}
this.quickSort(input, 0, input.length-1, k);
ArrayList<Integer> result = new ArrayList<>();
for (int i=0; i<k; i++) {
result.add(input[i]);
}
return result;
}
private void quickSort(int[] array, int startIndex, int endIndex, int k) {
int pivotIndex = this.partition(array, startIndex, endIndex);
if (k-1 < pivotIndex) {
quickSort(array, startIndex, pivotIndex-1, k);
} else if (k-1 > pivotIndex) {
quickSort(array, pivotIndex+1, endIndex, k);
} else {
return;
}
}
private void swap(int[] array, int src, int des) {
int temp = array[src];
array[src] = array[des];
array[des] = temp;
}
private int partition(int[] array, int startIndex, int endIndex) {
int mark = startIndex;
int pivot = startIndex;
for (int i=startIndex+1; i<=endIndex; i++) {
if (array[i] < array[pivot]) {
mark++;
this.swap(array, mark, i);
}
}
this.swap(array, pivot, mark);
return mark;
} - 归并
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
45public ArrayList<Integer> GetLeastNumbers_Solution(int [] input, int k) {
if (input.length == 0) {
return new ArrayList<Integer>();
}
this.sort(input, 0, input.length-1);
ArrayList<Integer> result = new ArrayList<>();
for (int i=0; i<k; i++) {
result.add(input[i]);
}
return result;
}
private void sort(int[] array, int startIndex, int endIndex) {
if (startIndex == endIndex) {
return;
}
int middleIndex = startIndex + (endIndex - startIndex) / 2;
sort(array, startIndex, middleIndex);
sort(array, middleIndex+1, endIndex);
merge(array, startIndex, endIndex, middleIndex);
}
private void merge(int[] array, int startIndex, int endIndex, int middleIndex) {
int[] mergeArray = new int[endIndex-startIndex+1];
int leftIndex = startIndex;
int rightIndex = middleIndex+1;
int mergeIndex = 0;
while (leftIndex<=middleIndex && rightIndex<=endIndex) {
if (array[leftIndex] < array[rightIndex]) {
mergeArray[mergeIndex++] = array[leftIndex++];
} else {
mergeArray[mergeIndex++] = array[rightIndex++];
}
}
while (leftIndex <= middleIndex) {
mergeArray[mergeIndex++] = array[leftIndex++];
}
while (rightIndex <= endIndex) {
mergeArray[mergeIndex++] = array[rightIndex++];
}
for (int i=0; i<mergeArray.length; i++) {
array[startIndex+i] = mergeArray[i];
}
}
数据流中的中位数
新加入的数插入到比第一个比它大的数前面
1 | private List<Integer> list = new ArrayList<>(); |