【经典排序算法】10. 基数排序

基数排序又是桶排序的改进,基数排序根据每个位数只包含10个数,开辟10个桶,通过对位数低到高地计数数组元素每个位数的数值(类似于计数排序),划分到对应桶中完成排序。对于十进制整数来说,基数排序是非常高效的。

public class Main {
    public static void main(String[] args) {
        int[] arr = {31, 5, 6, 2, 10, 17, 41};
        System.out.print("排序前:");
        arrPrint(arr);
        RadixSort(arr);
        System.out.print("排序后:");
        arrPrint(arr);
    }

    // 基数排序
    //
    private static void RadixSort(int[] arr) {
        radixsort(arr, arr.length - 1, maxBit(arr));
    }

    // maxBit函数:计算数组的最大位数
    // 具体方法为:先找到数组最大值max,之后反复除10,每次除不尽就在digit中累计+1,
    // 直至除max到max=0,此时digit的值就为数组arr的最大位数。
    // 本实例中max为41,最大位数digit=2。
    private static int maxBit(int[] arr) {
        int max = arr[0];
        for (int i = 0; i < arr.length; i++) {
            if (max < arr[i])
                max = arr[i];
        }
        int digit = 0;
        while (max != 0) {
            digit++;
            max /= 10;
        }
        return digit;
    }

    // 基数排序函数
    // 输入为待排序数组arr,数组arr的最大位数digit,在本示例中digit=2
    //
    // 第一层for循环,从位数1遍历到arr存在的最大位数digit,如当digit=2时,d遍历1和2.
    //
    // 第二层第一个for循环,计数数组count所有元素归零。
    //
    // 第二层第二个for循环,从左到右遍历数组arr,遍历元素为arr[i],使用getDigit函数
    // 获取arr[i]元素第d位的数值,记为j(比如31的第1位数值为1,第2位的数值为3)。
    // 那么j必然分布为0-9,正好是count的索引,则count[j]加1。类似于计数排序,使用count数组
    // 记录数组arr中第d位数字出现的数值j的出现次数,比如d=1时,遍历到31,j=1,则count[1]位置加1。
    //
    // 第二层第三个for循环,类似于计数排序,遍历计数数组count,将count的前一个位置计数
    // 结果累加到后一个位置计数结果中,此时count的元素,表示在arr中,小于等于对应第d位数值
    // 为count当前索引的元素的数量,(如count[1]=3,表示arr中,小于等于第d位数值为1的元素
    // 的数量为3)。
    //
    // 第二层第四个for循环,如同计数排序,倒序遍历arr的元素arr[i],getDigit函数获取arr[i]
    // 的第d位数值j,以j为索引获取count中的计数值count[j],那么count[j] - 1表示arr中,
    // 除了自己(arr[i])以外的,第d位数值小于等于j的元素的个数。
    // 第d位数值小于等于j的元素的个数为count[j] - 1,那么arr[i]在排序后数组中就应该排名
    // 第count[j] - 1位,所以直接bucket[count[j] - 1] = arr[i]即可。排序完count[j]位置累减
    //
    // 第二层第五个for循环,将桶中元素按顺序传arr。
    private static void radixsort(int[] arr, int len, int digit) {
        final int radix = 10;
        int j = 0;
        int[] count = new int[radix];
        int[] bucket = new int[len + 1];

        for (int d = 1; d <= digit; d++) {
            for (int i = 0; i < radix; i++) {
                count[i] = 0;
            }

            for (int i = 0; i <= len; i++) {
                j = getDigit(arr[i], d);
                count[j]++;
            }

            for (int i = 1; i < radix; i++) {
                count[i] += count[i - 1];
            }

            for (int i = len; i >= 0; i--) {
                j = getDigit(arr[i], d);
                bucket[count[j] - 1] = arr[i];
                count[j]--;
            }

            j = 0;
            for (int i = 0; i <= len; i++, j++) {
                arr[i] = bucket[j];
            }
        }
    }

    // getDigit函数
    // 返回数字x中第d位的数值.比如x=31, d=1时,返回1(31中第1位的数值为1)
    private static int getDigit(int x, int d) {
        return (x / ((int) Math.pow(10, d - 1))) % 10;
    }

    // 辅助函数:将int[] 打印出来
    private static void arrPrint(int[] arr) {
        StringBuilder str = new StringBuilder();
        str.append("[");
        for (int v : arr) {
            str.append(v + ", ");
        }
        str.delete(str.length() - 2, str.length());
        str.append("]");
        System.out.println(str.toString());
    }
}

本实例的动画演示如下:

在这里插入图片描述

猜你喜欢

转载自blog.csdn.net/fisherish/article/details/114686427