数据结构-排序之基数排序(使用java代码实现)

Posted 没谱的曲

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了数据结构-排序之基数排序(使用java代码实现)相关的知识,希望对你有一定的参考价值。

前言

最近在学习数据结构的排序算法时,学到了基数排序。对于基数排序的算法的具体实现过程有了一定了解,但在具体实现的时候出现了一些小问题。在和同学讨论和查阅资料过后打算使用java代码将其实现出来。

基数排序

基数排序是桶排序(或箱排序)的优化算法,解决了桶排序对于差值过大的列表排序时造成的大量空间冗余。
基数排序是根据数位进行排序的,它从每个数的个位开始,将对应的数存入相应的桶中,然后从桶中顺序取出数进行排序,排序完成之后,进一位,从十位开始,重复上述步骤,直至将最大位数的数的最高位进行排序完毕,最后获得的数就会是顺序的了。

具体代码

排序部分

因为每一位的排序都是重复的,在不确定最大位数为多少时,我们无法确定需要进行几次存放数据的重复操作,所以嵌套了一个循环,循环进行最大位数的排序次数。

public static void radixSort(int[] arr) {
        //先求出数组中最大的数
        int max = arr[0];
        for (int i = 1; i < arr.length; i++) {
            if (arr[i] > max) {
                max = arr[i];
            }
        }
        //得到最大数是几位数
        int maxLength = (max + "").length();

        //定义一个二维数组,表示10个桶,每个桶就是一个数组
        //为了在放入数时放置数据溢出,我们每个桶的大小为arr.length,即每个桶最多放进数组里的所有元素
        int[][] bucket = new int[10][arr.length];

        //定义一个一维数组来记录各个桶的每次放入的数据个数
        int[] bucketElementCount = new int[10];

        //根据最大的数有多少位进行遍历多少遍
        for (int count = 0; count < maxLength; count++) {
            //第count+1轮:针对每个元素的个、十、百、千。。。位进行排序处理
            for (int j = 0; j < arr.length; j++) {
                int digit = (int) Math.pow(10, count);
                //取出每个元素的个、十、百、千。。。位进行排序处理
                int digitOfElement = arr[j] / (1 * digit) % 10;
                //放入到个位数对应的桶中
                bucket[digitOfElement][bucketElementCount[digitOfElement]] = arr[j];
                bucketElementCount[digitOfElement]++;
            }
            //按照一维数组的下标(即桶的顺序)依次取出数据,放入到原来数组
            int index = 0;
            //遍历每一个桶,并将桶中数据放入到原数组
            for (int k = 0; k < bucketElementCount.length; k++) {
                //如果桶中有数据,才放入原来数组
                if (bucketElementCount[k] != 0) {
                    for (int m = 0; m < bucketElementCount[k]; m++) {
                        arr[index++] = bucket[k][m];
                    }
                }
                //第一轮处理后,需将bucketElementCount[k]=0
                bucketElementCount[k] = 0;
            }
            System.out.println("第" + (count+1) + "轮, arr=" + Arrays.toString(arr));
        }

    }

输入数组

java中不能直接通过手动输入整个数组,所以我写了一个方法,实现调用方法输入数组,并且可以控制输入的数组长度。

 public static int[] scanArray() {
        Scanner scanner = new Scanner(System.in);
        System.out.println("\\r请输入需要排序的数组的长度:");
        int length = scanner.nextInt();
        System.out.println("\\r请输入每一位需要排序的数值:");
        // 定义一个数组用于保存输入的数据
        int[] arr = new int[length];
        int digit = 1;
        int maxDigit = digit;
        for (int i = 0; i < arr.length; i++) {
            // 把数据按照从0开始的下标存入arr 数组
            arr[i] = scanner.nextInt();
        }
        return arr;
    }

实现排序

方法都封装完成后,我们就可以直接调用方法进行基数排序了

    public static void main(String[] args) {
        int[] arr = scanArray();
        radixSort(arr);
    }

运行结果:
在这里插入图片描述

总结

基数排序的时间复杂度一定是线性级别的,但是虽然是线性级别的,但是有一个系数的,这个系数就是最大元素的位数K,所以时间复杂度应该是O(n*k),空间复杂度主要就是取决于链表的数量以及序列元素的数量,所以空间复杂度为O(n+k)

以上是关于数据结构-排序之基数排序(使用java代码实现)的主要内容,如果未能解决你的问题,请参考以下文章

数据结构-排序之基数排序(使用java代码实现)

数据结构之排序算法Java实现—— 基数排序法

数据结构Java版之基数排序

为啥我的基数排序 JAVA 实现比快速排序慢?

算法-java代码实现基数排序

重温基础算法内部排序之基数排序法