1.基本排序算法:(5(3+2)+3)
5:
3-1.冒泡(稳定),
// 冒泡排序,a 表示数组,n 表示数组大小
public void bubbleSort(int[] a, int n) {
if (n <= 1) return;
for (int i = 0; i < n; ++i) {
// 提前退出冒泡循环的标志位
boolean flag = false;
for (int j = 0; j < n - i - 1; ++j) {
if (a[j] > a[j+1]) { // 交换
int tmp = a[j];
a[j] = a[j+1];
a[j+1] = tmp;
flag = true; // 表示有数据交换
}
}
if (!flag) break; // 没有数据交换,提前退出
}
}
3-2.插入(稳定),
// 插入排序,a 表示数组,n 表示数组大小
public void insertionSort(int[] a, int n) {
if (n <= 1) return;
for (int i = 1; i < n; ++i) {
int value = a[i];
int j = i - 1;
// 查找插入的位置
for (; j >= 0; --j) {
if (a[j] > value) {
a[j+1] = a[j]; // 数据移动
} else {
break;
}
}
a[j+1] = value; // 插入数据
}
}
(冒泡和插入虽然在逆序度上是相同的,但是,冒泡排序需要3个赋值操作,而插入排序只需要1个)
希尔:https://zh.wikipedia.org/wiki/%E5%B8%8C%E5%B0%94%E6%8E%92%E5%BA%8F
(衍生于插入排序,在插入上优化)
public static void shellSort(int[] array) {
int number = array.length / 2;
int i;
int j;
int temp;
while (number >= 1) {
for (i = number; i < array.length; i++) {
temp = array[i];
j = i - number;
while (j >= 0 && array[j] < temp) {
array[j + number] = array[j];
j = j - number;
}
array[j + number] = temp;
}
number = number / 2;
}
}
3-3.选择(类插入,不稳定)(O(n^2)),
以上适合于小规模数据的排序,以下两种适合大规模数据排序
2-1快排:分治思想, 非稳定排序
package sorts;
/**
* Created by wangzheng on 2018/10/16.
*/
public class QuickSort {
// 快速排序,a是数组,n表示数组的大小
public static void quickSort(int[] a, int n) {
quickSortInternally(a, 0, n-1);
}
// 快速排序递归函数,p,r为下标
private static void quickSortInternally(int[] a, int p, int r) {
if (p >= r) return;
int q = partition(a, p, r); // 获取分区点
quickSortInternally(a, p, q-1);
quickSortInternally(a, q+1, r);
}
private static int partition(int[] a, int p, int r) {
int pivot = a[r];
int i = p;
for(int j = p; j < r; ++j) {
if (a[j] < pivot) {
int tmp = a[i];
a[i] = a[j];
a[j] = tmp;
++i;
}
}
int tmp = a[i];
a[i] = a[r];
a[r] = tmp;
System.out.println("i=" + i);
return i;
}
}
2-2归并(O(nlogn)):非原地排序算法,分治思想
package sorts;
/**
* Created by wangzheng on 2018/10/16.
*/
public class MergeSort {
// 归并排序算法, a是数组,n表示数组大小
public static void mergeSort(int[] a, int n) {
mergeSortInternally(a, 0, n-1);
}
// 递归调用函数
private static void mergeSortInternally(int[] a, int p, int r) {
// 递归终止条件
if (p >= r) return;
// 取p到r之间的中间位置q,防止(p+r)的和超过int类型最大值
int q = p + (r - p)/2;
// 分治递归
mergeSortInternally(a, p, q);
mergeSortInternally(a, q+1, r);
// 将A[p...q]和A[q+1...r]合并为A[p...r]
merge(a, p, q, r);
}
private static void merge(int[] a, int p, int q, int r) {
int i = p;
int j = q+1;
int k = 0; // 初始化变量i, j, k
int[] tmp = new int[r-p+1]; // 申请一个大小跟a[p...r]一样的临时数组
while (i<=q && j<=r) {
if (a[i] <= a[j]) {
tmp[k++] = a[i++]; // i++等于i:=i+1
} else {
tmp[k++] = a[j++];
}
}
// 判断哪个子数组中有剩余的数据
int start = i;
int end = q;
if (j <= r) {
start = j;
end = r;
}
// 将剩余的数据拷贝到临时数组tmp
while (start <= end) {
tmp[k++] = a[start++];
}
// 将tmp中的数组拷贝回a[p...r]
for (i = 0; i <= r-p; ++i) {
a[p+i] = tmp[i];
}
}
}
3:
线性排序(非基于比较的算法,都涉及元素之间的比较操作):
3-1 Bucket sort:桶排序,比较适合用在外部排序中
https://blog.youkuaiyun.com/YinhJiang/article/details/80397415(桶排序:简易版本)
https://blog.youkuaiyun.com/LG1259156776/article/details/48803043(桶排序:标准版本)
3-2 Counting sort:计数排序,(桶排序的一种特殊情况)范围不大时,且非负整数
https://blog.youkuaiyun.com/qq_34328833/article/details/53442582(计数排序:空间换时间的代表)
3-3 Radix sort基数排序:
对要排序的数据是有要求的,需要可以分割出独立的“位”来比较,而且为之间有递进关系,如果a数据的高位比b数据大,那剩下的低位就不重要了。此外,每一位的数据范围不能太大,要可以用线性排序算法来排序,否则,基数排序的时间复杂度就没办法做到O(n).
https://www.cnblogs.com/skywang12345/p/3603669.html(基数排序:把之前需要O(n^2)转为O(n),优化了太多)
资源来源于各位博客共享,且有极客时间的资源总结