其中不理解时间复杂度得可以看一下我的时间复杂度的文章,有助于理解
代码git地址:https://github.com/gaoyeming/sort-algorithm.git
1,排序的定义
对一序列对象或者数组根据某个关键字进行排序
2、术语说明
稳定: 如果a原本在b前面,而a=b,排序之后a仍然在b的前面;
不稳定: 如果a原本在b的前面,而a=b,排序之后a可能会出现在b的前面;
内排序: 所有排序操作都在内存中完成;
外排序: 由于数据太大,因此把数据放在磁盘中,而排序通过磁盘和内存的数据传输才能进行;
时间复杂度: 一个算法执行所耗费的时间。
空间复杂度: 运行完一个程序所需内存的大小。
非线性时间比较类排序
:通过比较来决定元素间的相对次序,由于其时间复杂度不能突破O(nlogn),因此称为非线性时间比较类排序。线性时间非比较类排序
:不通过比较来决定元素间的相对次序,它可以突破基于比较排序的时间下界,以线性时间运行,因此称为线性时间非比较类排序。5、比较和非比较的区别
一、冒泡排序
百度百科解释: 冒泡排序是一种简单的排序算法。它重复地走访过要排序的数列,一次比较两个元素,如果他们的顺序错误就把它们交换过来。走访数列的工作是重复地进行直到没有再需要交换,也就是说该数列已经排序完成。这个算法的名字由来是因为越小的元素会经由交换慢慢”浮“到数列的顶端。
/**
*冒泡排序
*@param arry 需要排序的int数组
*@return 排序后的数组
*/
public static int[] bubbleSort(int[] arry) {
for (int i = 0; i < arry.length; i++) {
for (int j = i + 1; j < arry.length; j++) {
if (arry[j] < arry[i]) {
int tmp = arry[j];
arry[j] = arry[i];
arry[i] = tmp;
}
}
}
return arry;
}
二、选择排序
百度百科解释: 选择排序(Selection-sort)是一种简单直观的排序算法。它的工作原理:首先在未排序列中找到最小或者最大的元素,存放到排序序列的起始位置,然后,再从剩余未排序元素中继续寻找最小或者最大元素,然后存放到已排序的序列的末尾。以此类推,直到所有元素均排序完毕。
/**
*选择排序
*@param array 需要排序的int数组
*@return 排序后的数组
*/
public static int[] selectionSort(int[] array) {
for (int i = 0; i < array.length; i++) {
int minIndex = i;
for (int j = i; j < array.length; j++) {
if (array[j] < array[minIndex]) {
minIndex = j;
}
}
if (minIndex != i) {
int tmp = array[minIndex];
array[minIndex] = array[i];
array[i] = tmp;
}
}
return array;
}
三、插入排序
百度百科解释: 插入排序,一般也被称为直接插入排序。对于少量元素的排序,它是一个有效的算法。插入排序是一种最简单的排序方法,它的基本思想是将一个记录插入到已经排好序的有序表中,从而一个新的、记录数增1的有序表。在其实现过程使用双层循环,外层循环对除了第一个元素之外的所有元素,内层循环对当前元素前面有序表进行待插入位置查找,并进行移动。
/**
*插入排序
*@param array 需要排序的int数组
*@return 排序后的数组
*/
public static int[] insertionSort(int[] array) {
for (int i = 1; i < array.length; i++) {
int current = array[i];
int preIndex = i - 1;
while (preIndex >= 0 && current < array[preIndex] ) {
array[preIndex + 1] = array[preIndex];
preIndex--;
}
array[preIndex + 1] = current;
}
return array;
}
四、希尔排序
百度百科解释: 希尔排序(Shell’s Sort)是插入排序的一种又称“缩小增量排序”(Diminishing Increment Sort),是直接插入排序算法的一种更高效的改进版本。希尔排序是非稳定排序算法。该方法因D.L.Shell于1959年提出而得名。
希尔排序是把记录按下标的一定增量分组,对每组使用直接插入排序算法排序;随着增量逐渐减少,每组包含的关键词越来越多,当增量减至1时,整个文件恰被分成一组,算法便终止。
/**
*希尔排序
*@param array 需要排序的int数组
*@return 排序后的数组
*/
public static int[] shellSort(int[] array) {
int length = array.length;
int gap = length / 2;
while (gap > 0) {
for (int i = gap; i < length; i++) {
int current = array[i];
int preIndex = i - gap;
while ( preIndex >= 0 && current < array[preIndex]) {
array[preIndex + gap] = array[preIndex];
preIndex -= gap;
}
array[preIndex + gap] = current;
}
gap /= 2;
}
return array;
}
五、归并排序
百度百科解释: 归并排序(MERGE-SORT)是建立在归并操作上的一种有效的排序算法,该算法是采用分治法(Divide and Conquer)的一个非常典型的应用。将已有序的子序列合并,得到完全有序的序列;即先使每个子序列有序,再使子序列段间有序。若将两个有序表合并成一个有序表,称为二路归并。归并排序是一种稳定的排序方法。
/**
*归并排序
*@param array 需要排序的int数组
*@return 排序后的数组
*/
public static int[] mergeSort(int[] array) {
if (array.length < 2) {
return array;
}
int mid = array.length / 2;
int[] left = Arrays.copyOfRange(array, 0, mid);
int[] right = Arrays.copyOfRange(array, mid, array.length);
return merge(mergeSort(left), mergeSort(right));
}
/**
*归并排序——将两段排序好的数组结合成一个排序数组
*@param left
*@param right
*@return
*/
private static int[] merge(int[] left, int[] right) {
int[] result = new int[left.length + right.length];
for (int index = 0, i = 0, j = 0; index < result.length; index++) {
if (i >= left.length) {
result[index] = right[j++];
} else if (j >= right.length) {
result[index] = left[i++];
} else if (left[i] > right[j]) {
result[index] = right[j++];
} else {
result[index] = left[i++];
}
}
return result;
}
六、快速排序
百度百科解释: 快速排序(Quicksort)是对冒泡排序的一种改进。
快速排序由C. A. R. Hoare在1960年提出。它的基本思想是:通过一趟排序将要排序的数据分割成独立的两部分,其中一部分的所有数据都比另外一部分的所有数据都要小,然后再按此方法对这两部分数据分别进行快速排序,整个排序过程可以递归进行,以此达到整个数据变成有序序列。
/**
*快速排序
*@param array 需要排序的int数组
*@param startIndex 开始的索引位置
*@param endIndex 结束的索引位置
*@return 排序后的数组
*/
public static int[] quickSort(int[] array, int startIndex, int endIndex) {
if (array.length < 1 || startIndex < 0 || endIndex >= array.length || startIndex > endIndex) {
return null;
}
int pivotIndex = partition(array, startIndex, endIndex);
if (pivotIndex > startIndex) {
quickSort(array, startIndex, pivotIndex - 1);
}
if (pivotIndex < endIndex) {
quickSort(array, pivotIndex + 1, endIndex);
}
return array;
}
/**
*快速排序算法——partition
*@param array 分区的数组
*@param startIndex 开始的索引位置
*@param endIndex 结束的索引位置
*@return 返回分区过后的数组
*/
private static int partition(int[] array, int startIndex, int endIndex) {
//随机获取一个基准(需要保证在数组内)
int pivotIndex = (int) (startIndex + Math.random() * (endIndex - startIndex + 1));
int i = startIndex;
while (i <= endIndex) {
if (i <= pivotIndex) {
if (array[i] > array[pivotIndex]) {
swap(array, i, pivotIndex);
pivotIndex = i;
}
} else {
if (array[i] < array[pivotIndex]) {
swap(array, i, pivotIndex);
int tmp = pivotIndex;
pivotIndex = i;
i = tmp;
}
}
i++;
}
return pivotIndex;
}
/**
*交换数组内两个元素
*@param array 源元素数组
*@param i 交换位置i
*@param j 交换位置j
*/
private static void swap(int[] array, int i, int j) {
int temp = array[i];
array[i] = array[j];
array[j] = temp;
}
七、堆排序
首先了解二叉树:
上图就是一个完全二叉树,其特点在于:
1,从作为第一层根开始,除了最后一层之外,第N层的元素个数都必须是2的N-1次方;第一层一个,第二层两个,第三层四个,以此类推
2,而最后一行的元素,都是紧贴在左边,换句话说,每一行的元素都是从最左边开始安放,两个元素之间不能有空闲;具备了这两个特点的树,就是一颗完全二叉树。
那么,完全二叉树与堆有什么关系呢?
我们假设有一颗完全二叉树,在满足作为二叉树的基础上,对于任意一个拥有父节点的子节点,其数值均不小于父节点的值;这样层层递推,就是根节点的值最小,这样的数称为小根堆
同理,又有一颗完全二叉树,对于任意一个子节点来说,均不大于其父节点的值,如此递推,就是根节点的值是最大的,这样的数称为大跟堆。
如上图,左边就是大根堆;右边则是小根堆,这里必须要注意一点,只要求子节点与父节点的关系,两个节点的大小关系与其左右位置没有任何关系。
备注: 一般升序采用大顶堆,降序采用小顶堆
百度百科解释: 堆排序(英语:Heapsort)是指利用堆这种数据结构所设计的一种排序算法。堆是一个近似完全二叉树的结构,并同时满足堆积的性质:即子结点的键值或索引总是小于(或者大于)它的父节点。
步骤二、将堆顶元素与末尾元素进行交换,使末尾元素最大。然后继续调整堆,再将堆顶元素与末尾元素交换,得到第二大元素,如此反复进行交换、重建、交换。
/**
*堆排序
*@param array 需要排序的int数组
*@return 排序后的数组
*/
private static int len;
public static int[] heapSort(int[] array){
len = array.length;
if (len < 1) {
return array;
}
//1.构建一个最大堆
buildMaxHeap(array);
//2.循环将堆首位(最大值)与末位交换,然后在重新调整最大堆
while (len > 0) {
swap(array, 0, len - 1);
len--;
adjustHeap(array, 0);
}
return array;
}
/**
*建立最大堆
*@param array 需要构建的数组
*/
private static void buildMaxHeap(int[] array) {
//从最后一个非叶子节点开始向上构造最大堆
for (int i = (len-1) / 2; i >= 0; i--) {
adjustHeap(array ,i);
}
}
/**
*调整使之成为最大堆
*@param array
*@param i
*/
private static void adjustHeap(int[] array, int i) {
int maxIndex = i;
//如果有左子树,且左子树大于父节点,则将最大指针指向左子树
if (i * 2 < len && array[i * 2] > array[maxIndex]) {
maxIndex = i * 2;
}
//如果有右子树,且右子树大于父节点,则将最大指针指向右子树
if (i * 2 + 1 < len && array[i * 2 + 1] > array[maxIndex]) {
maxIndex = i * 2 + 1;
}
//如果父节点不是最大值,则将父节点与最大值交换,并且递归调整与父节点交换的位置。
if (maxIndex != i) {
swap(array, maxIndex, i);
adjustHeap(array, maxIndex);
}
}
/**
*交换数组内两个元素
*@param array 源元素数组
*@param i 交换位置i
*@param j 交换位置j
*/
private static void swap(int[] array, int i, int j) {
int temp = array[i];
array[i] = array[j];
array[j] = temp;
}
八、计数排序
百度百科解释: 计数排序是一个非基于比较的排序算法,该算法于1954年由 Harold H. Seward 提出。它的优势在于在对一定范围内的整数排序时,它的复杂度为Ο(n+k)(其中k是整数的范围),快于任何比较排序算法。当然这是一种牺牲空间换取时间的做法,而且当O(k)>O(nlog(n))的时候其效率反而不如基于比较的排序(基于比较的排序的时间复杂度在理论上的下限是O(nlog(n)), 如归并排序,堆排序)
/**
*计数排序(仅支持整数排序)
*@param array 需要排序的int数组
*@return 排序后的数组
*/
public static int[] countingSort(int[] array) {
//首先取出array数组中最大,最小的两个元素
int maxElement = array[0];
int minElement = array[0];
for (int anArray : array) {
if (anArray > maxElement) {
maxElement = anArray;
}
if (anArray < minElement) {
minElement = anArray;
}
}
//定义一个额外空间数组;长度为最大元素与最小元素之间存在的整数个数
int[] extraArray = new int[maxElement - minElement + 1];
Arrays.fill(extraArray, 0);
for (int anArray : array) {
extraArray[anArray - minElement]++;
}
//反向填充目标数组
int index = 0, i = 0;
while (index < array.length) {
if (extraArray[i] != 0) {
array[index] = i + minElement;
extraArray[i]--;
index++;
} else {
i++;
}
}
return array;
}
九、桶排序
百度百科解释: 桶排序 (Bucket sort)或所谓的箱排序,是一个排序算法,工作的原理是将数组分到有限数量的桶子里。每个桶子再个别排序(有可能再使用别的排序算法或是以递归方式继续使用桶排序进行排序)。桶排序是鸽巢排序的一种归纳结果。当要被排序的数组内的数值是均匀分配的时候,桶排序使用线性时间(Θ(n))。但桶排序并不是 比较排序,他不受到 O(n log n) 下限的影响。
/**
*桶排序
*桶排序中:无序数组有个要求,就是成员隶属于固定(有限的)的区间,如范围为0-9
*@param array 需要排序的int数组
*@param bucketSize 桶的大小
*@return 排序后的数组
*/
public static ArrayList bucketSort(int[] array, int bucketSize) {
//首先取出array数组中最大,最小的两个元素
int maxElement = array[0];
int minElement = array[0];
for (int anArray : array) {
if (anArray > maxElement) {
maxElement = anArray;
}
if (anArray < minElement) {
minElement = anArray;
}
}
//计算出桶的数量
int bucketCount = (maxElement - minElement) / bucketSize + 1;
//定义桶
List> buckets = new ArrayList<>(bucketCount);
for (int i = 0; i < bucketCount; i++) {
buckets.add(new ArrayList<>());
}
//装载桶
for (int anArray : array) {
//根据值计算出所在的桶索引
int bucketIndex = anArray / bucketSize;
buckets.get(bucketIndex).add(anArray);
}
//对每个桶利用快排(其他方式排序也是可以的)进行排序
ArrayList sortedList = new ArrayList<>(bucketCount);
for (List bucket : buckets) {
int[] bucketArray = new int[bucket.size()];
for (int i = 0; i < bucket.size(); i++) {
bucketArray[i] = bucket.get(i);
}
sortedList.add(quickSort(bucketArray, 0, bucketArray.length - 1));
}
return sortedList;
}
十、基数排序
百度百科解释: 基数排序(radix sort)属于“分配式排序”(distribution sort),又称“桶子法”(bucket sort)或bin sort,顾名思义,它是透过键值的部份资讯,将要排序的元素分配至某些“桶”中,藉以达到排序的作用,基数排序法是属于稳定性的排序,其时间复杂度为O (nlog®m),其中r为所采取的基数,而m为堆数,在某些时候,基数排序法的效率高于其它的稳定性排序法。
/**
*基数排序
*@param array 需要排序的int数组
*@return 排序后的数组
*/
public static int[] radixSort(int[] array) {
//先获取最大数
int maxElement = array[0];
for (int anArray : array) {
if (anArray > maxElement) {
maxElement = anArray;
}
}
//计算其对应的位数
int maxDigit = 0;
while (maxElement != 0) {
maxElement /= 10;
maxDigit++;
}
//定义长度为10的桶radix
List> radixList = new ArrayList<>(10);
for (int i = 0; i < 10; i++) {
radixList.add(new ArrayList<>());
}
//从最低位开始取每个位组成radix数组;
int mod = 10, div = 1;
for(int i=0;i