Java实现十大排序算法,配合动态图片

Java实现排序算法

其中不理解时间复杂度得可以看一下我的时间复杂度的文章,有助于理解

十大排序算法

  • 1、冒泡排序
  • 2、选择排序
  • 3、插入排序
  • 4、希尔排序
  • 5、归并排序
  • 6、快速排序
  • 7、堆排序
  • 8、计数排序
  • 9、桶排序
  • 10、基数排序

代码git地址:https://github.com/gaoyeming/sort-algorithm.git

排序算法说明

  • 1,排序的定义
    对一序列对象或者数组根据某个关键字进行排序

  • 2、术语说明
    稳定: 如果a原本在b前面,而a=b,排序之后a仍然在b的前面;
    不稳定: 如果a原本在b的前面,而a=b,排序之后a可能会出现在b的前面;
    内排序: 所有排序操作都在内存中完成;
    外排序: 由于数据太大,因此把数据放在磁盘中,而排序通过磁盘和内存的数据传输才能进行;
    时间复杂度: 一个算法执行所耗费的时间。
    空间复杂度: 运行完一个程序所需内存的大小。

  • 3、算法总结
    在这里插入图片描述
    图片名词解释:

    • n:数据规模
    • k:”桶“的个数
    • In-place:占用常数内存,不占用额外内存
    • Out-place:占用额外内存
  • 4、算法分类
    在这里插入图片描述

    • 非线性时间比较类排序 :通过比较来决定元素间的相对次序,由于其时间复杂度不能突破O(nlogn),因此称为非线性时间比较类排序。
    • 线性时间非比较类排序 :不通过比较来决定元素间的相对次序,它可以突破基于比较排序的时间下界,以线性时间运行,因此称为线性时间非比较类排序。
  • 5、比较和非比较的区别

    • 常见的快速排序、归并排序、堆排序、冒泡排序等属于比较排序。在排序的最终结果里,元素之间的次序依赖于它们之间的比较。每个数都必须和其他数进行比较,才能确定自己的位置。
    • 在冒泡排序之类的排序中,问题规模为n,又因为需要比较n次,所以平均的时间复杂度为O(n^2)。在归并排序、快速排序之类的排序中,问题规模通过分治法消减为logN次,所以时间复杂度平均为O(nlogn)。比较排序的优势是,适用于各种规模的数据,也不在乎数据的分布,都能进行排序。可以说,比较排序适用于一切需要排序的情况。
    • 计数排序,基数排序,桶排序则属于非比较排序。非比较排序是通过确定每个元素之前,应该有多少个元素来排序。针对数组arr,计算arr[i]之前有多少个元素,则唯一确定了arr[i]在排序后数组中的位置。非比较排序只要确定每个元素之前的已有的元素个数即可,所有一次遍历即可解决。算法时间复杂度O(n).
      非比较排序时间复杂度低,但由于非比较排序需要占用空间来确定唯一位置。所以对数据规模和数据分布有一定的要求。

下面开始真正实现排序及理解对应思想

一、冒泡排序

百度百科解释: 冒泡排序是一种简单的排序算法。它重复地走访过要排序的数列,一次比较两个元素,如果他们的顺序错误就把它们交换过来。走访数列的工作是重复地进行直到没有再需要交换,也就是说该数列已经排序完成。这个算法的名字由来是因为越小的元素会经由交换慢慢”浮“到数列的顶端。

  • 1.1、算法描述
    • 比较相邻的元素。如果第一个比第二个大,就交换他们两个;
    • 对每一对相邻元素作同样的工作,从开始第一对到结尾的最后一对,这样在最后的元素应该会是最大的数;
    • 针对所有的元素重复以上的步骤,除了最后一个;
    • 重复以上步骤,直到排序完成
  • 1.2、动图演示
    在这里插入图片描述
  • 1.3、代码实现
/**
*冒泡排序
*@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;
}
  • 1.4、时间复杂度
    • 最佳情况:T(n)=O(n)
    • 最差情况:T(n)=O(n^2)
    • 平均情况:T(n)=O(n^2)

二、选择排序

百度百科解释: 选择排序(Selection-sort)是一种简单直观的排序算法。它的工作原理:首先在未排序列中找到最小或者最大的元素,存放到排序序列的起始位置,然后,再从剩余未排序元素中继续寻找最小或者最大元素,然后存放到已排序的序列的末尾。以此类推,直到所有元素均排序完毕。

  • 2.1、算法描述
    n个记录的直接选择排序可经过n-1趟直接选择排序得到有序的结果。具体算法描述如下:
    • 初始状态,无序区为R[1…n],有序区为空
    • 第一趟排序(i=1,2,3…n-1)开始时,当前有序区和无序区分别为R[1…i-1]和R[i…n]。该趟排序从当前无序区中选出关键字最小的记录R[k],将它与无序区的第一个记录R交换,使得R[1…i]和R[i+1…n]分别变为记录个数增加1个的新有序区和记录个数减少一个的新无序区
    • n-1趟结束,数组有序化了。
  • 2.2、动态演示
    在这里插入图片描述
  • 2.3、代码实现
/**
*选择排序
*@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;
}
  • 2.4、时间复杂度
    • 最佳情况:T(n)=O(n^2)
    • 最差情况T(n)=O(n^2)
    • 平均情况T(n)=O(n^2)

三、插入排序

百度百科解释: 插入排序,一般也被称为直接插入排序。对于少量元素的排序,它是一个有效的算法。插入排序是一种最简单的排序方法,它的基本思想是将一个记录插入到已经排好序的有序表中,从而一个新的、记录数增1的有序表。在其实现过程使用双层循环,外层循环对除了第一个元素之外的所有元素,内层循环对当前元素前面有序表进行待插入位置查找,并进行移动。

  • 3.1、算法描述
    一般来说,插入排序都采用in-place在数组上实现。具体算法描述如下:
    • 从第一个元素开始,该元素可以认为已经被排序;
    • 取出下一个元素,在已经排序的元素序列中从后向前扫描;
    • 如果该元素(已排序)大于新元素,将该元素移到下一位置;
    • 重复步骤3,直到找到已排序的元素小于或者等于新元素的位置;
    • 将新元素插入到该位置后;
    • 重复步骤2~5
  • 3.2、动图演示
    在这里插入图片描述
  • 3.3、代码实现
/**
*插入排序
*@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;
}
  • 3.4、时间复杂度
    • 最佳情况:T(n)=O(n)
    • 最坏情况T(n)=O(n^2)
    • 平均情况T(n)=O(n^2)

四、希尔排序

百度百科解释: 希尔排序(Shell’s Sort)是插入排序的一种又称“缩小增量排序”(Diminishing Increment Sort),是直接插入排序算法的一种更高效的改进版本。希尔排序是非稳定排序算法。该方法因D.L.Shell于1959年提出而得名。
希尔排序是把记录按下标的一定增量分组,对每组使用直接插入排序算法排序;随着增量逐渐减少,每组包含的关键词越来越多,当增量减至1时,整个文件恰被分成一组,算法便终止。

  • 4.1、算法描述
    我们来看下希尔排序的基本步骤,在此我们选择增量gap=length/2,缩小增量继续以gap = gap/2的方式,这种增量选择我们可以用一个序列来表示,{n/2,(n/2)/2…1},称为增量序列。希尔排序的增量序列的选择与证明是个数学难题,我们选择的这个增量序列是比较常用的,也是希尔建议的增量,称为希尔增量,但起始这个增量序列不是最优的。此处我们做示例使用希尔增量。
    先将整个待排序的记录序列分割成为若干子序列分别进行直接插入排序,具体算法描述:
    • 选择一个增量序列t1, t2, …, tk,其中ti>tj,tk=1;
    • 按增量序列个数k,对序列进行k趟排序
    • 每趟排序,根据对应的增量ti,将待排序列分割成若干长度为m的子序列,分别对各字表进行直接插入排序。仅增量因子为1时,整个序列作为一个表来处理,表长度即为整个序列的长度
  • 4.2、过程演示
    在这里插入图片描述
  • 4.3、代码实现
/**
*希尔排序
*@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;
}
  • 4.4、时间复杂度
    • 最佳情况:T(n) = O(n)
    • 最坏情况:T(n) = O(n^2)

五、归并排序

百度百科解释: 归并排序(MERGE-SORT)是建立在归并操作上的一种有效的排序算法,该算法是采用分治法(Divide and Conquer)的一个非常典型的应用。将已有序的子序列合并,得到完全有序的序列;即先使每个子序列有序,再使子序列段间有序。若将两个有序表合并成一个有序表,称为二路归并。归并排序是一种稳定的排序方法。

  • 5.1、算法描述
    • 把长度为n的输入序列分成两个长度为n/2的子序列;
    • 对这两个子序列分别采用归并排序;
    • 将两个排序好的子序列合并成一个最终的排序序列
  • 5.2、动图演示
    在这里插入图片描述
  • 5.3、代码实现
/**
*归并排序
*@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;
}
  • 5.4、时间复杂度
    • 最佳情况:T(n) = O(nlogn)
    • 最差情况:T(n) = O(nlogn)
    • 平均情况:T(n) = O(nlogn)

六、快速排序

百度百科解释: 快速排序(Quicksort)是对冒泡排序的一种改进。
快速排序由C. A. R. Hoare在1960年提出。它的基本思想是:通过一趟排序将要排序的数据分割成独立的两部分,其中一部分的所有数据都比另外一部分的所有数据都要小,然后再按此方法对这两部分数据分别进行快速排序,整个排序过程可以递归进行,以此达到整个数据变成有序序列。

  • 6.1、算法描述
    快速排序使用分治法来把一个串(list)分为两个子串(sub-list)。具体算法描述如下:
    • 从数列中挑出一个元素,称为“基准”(pivot);
    • 重新排序数列,所有元素比基准值小的摆放在基准的前面,所有元素比基准值大的摆放在基准的后面(相同的数可以放到任意一边)。在这个区分退出之后,该基准就处于数列的中间位置。这个称为分区(partition)操作。
    • 递归(recursive)把小于基准值元素的子数列和大于基准值元素的子数列排序
  • 6.2、动图演示
    在这里插入图片描述
  • 6.3、代码实现
/**
*快速排序
*@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;
}
  • 6.4、算法分析
    • 最佳情况:T(n) = O(nlogn)
    • 最差情况:T(n) = O(n2)
    • 平均情况:T(n) = O(nlogn)

七、堆排序

首先了解二叉树:
在这里插入图片描述
上图就是一个完全二叉树,其特点在于:
1,从作为第一层根开始,除了最后一层之外,第N层的元素个数都必须是2的N-1次方;第一层一个,第二层两个,第三层四个,以此类推
2,而最后一行的元素,都是紧贴在左边,换句话说,每一行的元素都是从最左边开始安放,两个元素之间不能有空闲;具备了这两个特点的树,就是一颗完全二叉树。
那么,完全二叉树与堆有什么关系呢?
我们假设有一颗完全二叉树,在满足作为二叉树的基础上,对于任意一个拥有父节点的子节点,其数值均不小于父节点的值;这样层层递推,就是根节点的值最小,这样的数称为小根堆
同理,又有一颗完全二叉树,对于任意一个子节点来说,均不大于其父节点的值,如此递推,就是根节点的值是最大的,这样的数称为大跟堆。

如上图,左边就是大根堆;右边则是小根堆,这里必须要注意一点,只要求子节点与父节点的关系,两个节点的大小关系与其左右位置没有任何关系。
备注: 一般升序采用大顶堆,降序采用小顶堆

百度百科解释: 堆排序(英语:Heapsort)是指利用堆这种数据结构所设计的一种排序算法。堆是一个近似完全二叉树的结构,并同时满足堆积的性质:即子结点的键值或索引总是小于(或者大于)它的父节点。

  • 7.1、算法描述
    • 将初始待排序关键字序列(R1, R2, … Rn)构建成大顶堆,此堆为初始的无序区;
    • 将堆顶元素R[1]与最后一个元素R[n]交换,此时得到新的无序区(R1, R2, … Rn-1)和新的有序区(Rn),且满足R[1, 2, … n-1]<=R[n];
    • 由于交换后新的堆顶R[1]可能违反堆的性质,因此需要对当前无序区(R1, R2, … Rn-1)调整为新堆,然后再次将R[1]与无序区最后一个元素交换,得到新的无序区(R1, R2 , … Rn-2)和新的有序区(Rn-1, Rn)。不断重复此过程直到有序区的元素个数为n-1,则整个排序过程完成
  • 7.2、简单实例
    给定一个列表array=[4,6,8,5,9],对其进行堆排序(使用大根堆)。
    步骤一、构造初始堆。将给定的无序序列构造成一个大顶堆
    • 1、假设给定无序序列结构如下:
      在这里插入图片描述
    • 2、此时我们从最后一个非叶子节点开始(叶节点自然不用调整,第一个非叶子结点arr.length/2-1=5/2-1=1,也就是下面的6结点,从左至右,从上至下进行调整)
      注意:我们把6和9比较交换之后,必须考量9这个节点对于其子节点会不会产生任何影响,因为其是叶子节点,所以不加考虑;但是,一定要熟练这种思维,写代码的时候就比较容易理解为什么会出现一次非常重要的交换了。
      在这里插入图片描述
    • 3、找到第二个非叶节点4,由于[4,9,8]中9元素最大,4和9交换
      在真正代码实现中,这时候4和9交换过后,必须考虑9所在的这个节点位置,因为其上的值变了,必须判断对其的两个子节点是否造成了影响。这么说不合适,实际上就是判断其作为根节点的那棵树,是都还满足大根堆的原则,每一次交换,都必须要循环把子数部分判别清楚。
      在这里插入图片描述
    • 这时交换导致了子根[4,5,6]出现混乱,继续调整,[4,5,6]中6最大,交换4和6.
      牢记上面说的规则,每次交换都要把改变了的那个节点所在的树重新判断一下,这里就用上了,4和9交换了,变动了的那棵子树就必须重新调整,一直调整到符合大根堆的规则为止。
      在这里插入图片描述
    • 此时,我们就将一个无序序列构造成了一个大顶堆。

步骤二、将堆顶元素与末尾元素进行交换,使末尾元素最大。然后继续调整堆,再将堆顶元素与末尾元素交换,得到第二大元素,如此反复进行交换、重建、交换。

  • 1、将对顶元素9和末尾元素4进行交换
    这里,必须说明一下,所谓的交换,实际上就是把最大值从树里面拿掉了,剩下参与到排序的树,其实只有总节点的个数减去拿掉节点个数了,所以下图用的是虚线。
    在这里插入图片描述
  • 2、重新调整结构,使其满足大根堆的定义
    在这里插入图片描述
  • 3、再将堆顶元素8与末尾元素5进行交换,得到第二大元素8
    在这里插入图片描述
  • 4、后续过程,继续调整,交换,如此反复进行,最终使得整个序列有序
    在这里插入图片描述
  • 7.3、动图演示
    在这里插入图片描述
  • 7.3、代码实现
/**
*堆排序
*@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;
}
  • 7.4、算法分析
    • 最佳情况:T(n) = O(nlogn)
    • 最差情况:T(n) = O(nlogn)
    • 平均情况:T(n) = O(nlogn)

八、计数排序

百度百科解释: 计数排序是一个非基于比较的排序算法,该算法于1954年由 Harold H. Seward 提出。它的优势在于在对一定范围内的整数排序时,它的复杂度为Ο(n+k)(其中k是整数的范围),快于任何比较排序算法。当然这是一种牺牲空间换取时间的做法,而且当O(k)>O(nlog(n))的时候其效率反而不如基于比较的排序(基于比较的排序的时间复杂度在理论上的下限是O(nlog(n)), 如归并排序,堆排序)

  • 8.1、算法描述
    • 找出待排序的数组中最大和最小的元素;
    • 统计数组中每个值为i的元素出现的次数,存入数组C的第i项;
    • 对所有的计数累加(从C中的第一个元素开始,每一项和前一项相加);
    • 反向填充目标数组:将每个元素i放在新数组的第C(i)项,每放一个元素就将C(i)减去1
  • 8.2、动图演示
    在这里插入图片描述
  • 8.3、代码实现
/**
*计数排序(仅支持整数排序)
*@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;
}
  • 8.4、算法分析
    当输入的元素是n 个0到k之间的整数时,它的运行时间是 O(n + k)。计数排序不是比较排序,排序的速度快于任何比较排序算法。由于用来计数的数组C的长度取决于待排序数组中数据的范围(等于待排序数组的最大值与最小值的差加上1),这使得计数排序对于数据范围很大的数组,需要大量时间和内存。
    • 最佳情况:T(n) = O(n+k)
    • 最差情况:T(n) = O(n+k)
    • 平均情况:T(n) = O(n+k)

九、桶排序

百度百科解释: 桶排序 (Bucket sort)或所谓的箱排序,是一个排序算法,工作的原理是将数组分到有限数量的桶子里。每个桶子再个别排序(有可能再使用别的排序算法或是以递归方式继续使用桶排序进行排序)。桶排序是鸽巢排序的一种归纳结果。当要被排序的数组内的数值是均匀分配的时候,桶排序使用线性时间(Θ(n))。但桶排序并不是 比较排序,他不受到 O(n log n) 下限的影响。

  • 9.1、算法描述
    • 人为设置一个BucketSize,作为每个桶所能放置多少个不同数值(例如当BucketSize==5时,该桶可以存放{1,2,3,4,5}这几种数字,但是容量不限,及可以存放100个3)
    • 遍历输入数据,并且把数据一个一个放到对应的桶里去;
    • 对每个不是空的桶进行排序,可以使用其它排序方法,也可以递归使用桶排序;
    • 从不是空的桶里把排好序的数据拼接起来
      注意:如果递归使用桶排序为各个桶排序,则当桶的数量为1时要手动减小BucketSize增加下一循环桶的数量,否则会陷入死循环,导致内存溢出。
  • 9.2、图片展示
    在这里插入图片描述
  • 9.3、代码实现
/**
*桶排序
*桶排序中:无序数组有个要求,就是成员隶属于固定(有限的)的区间,如范围为0-9
*@param array      需要排序的int数组
*@param bucketSize 桶的大小
*@return 排序后的数组
*/
public static ArrayList<int[]> 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<List<Integer>> 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<int[]> sortedList = new ArrayList<>(bucketCount);
    for (List<Integer> 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;
}
  • 9.4、算法分析
    桶排序最好情况下使用线性时间O(n),桶排序的时间复杂度,取决与对各个桶之间数据进行排序的时间复杂度,因为其它部分的时间复杂度都为O(n)。很显然,桶划分的越小,各个桶之间的数据越少,排序所用的时间也会越少。但相应的空间消耗就会增大。
    • 最佳情况:T(n) = O(n+k)
    • 最差情况:T(n) = O(n+k)
    • 平均情况:T(n) = O(n2)

十、基数排序

百度百科解释: 基数排序(radix sort)属于“分配式排序”(distribution sort),又称“桶子法”(bucket sort)或bin sort,顾名思义,它是透过键值的部份资讯,将要排序的元素分配至某些“桶”中,藉以达到排序的作用,基数排序法是属于稳定性的排序,其时间复杂度为O (nlog®m),其中r为所采取的基数,而m为堆数,在某些时候,基数排序法的效率高于其它的稳定性排序法。

  • 10.1、算法描述
    • 取得数组中的最大数,并取得位数;
    • arr为原始数组,从最低位开始取每个位组成radix数组;
    • 对radix进行计数排序(利用计数排序使用于小范围数的特点)
  • 10.2、动图展示
    在这里插入图片描述
  • 10.3、代码实现
/**
*基数排序
*@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<List<Integer>> 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<maxDigit;i++){
        for(int anArray : array){
            int num = (anArray % mod) / div;
            radixList.get(num).add(anArray);
        }

        int index = 0;
        for (int j = 0; j < radixList.size(); j++) {
            for (int k = 0; k < radixList.get(j).size(); k++) {
                array[index++] = radixList.get(j).get(k);
            }
            radixList.get(j).clear();
        }
        mod *= 10; div *= 10;
    }

    return array;
}
  • 10.4、算法分析
    • 最佳情况:T(n) = O(n * k)
    • 最差情况:T(n) = O(n * k)
    • 平均情况:T(n) = O(n * k)
    • 基数排序有两种方法:
      MSD 从高位开始进行排序 LSD 从低位开始进行排序
      基数排序 vs 计数排序 vs 桶排序

最后对于时间复杂度不理解的可以看一下我的对于时间复杂度理解的文章。如果你不想复制代码,可以直接从我的github上下载:github实现十大算法代码

展开阅读全文

没有更多推荐了,返回首页

©️2019 CSDN 皮肤主题: 大白 设计师: CSDN官方博客
应支付0元
点击重新获取
扫码支付

支付成功即可阅读