java实现基数排序(详细解释代码和逻辑)

简介: java实现基数排序(详细解释代码和逻辑)

基数排序是一种非比较型整数排序算法,其基本思想是将整数按位数切分,然后按每个位数分别排序。基数排序可以分为 LSD(Least Significant Digit,最低位优先)和 MSD(Most Significant Digit,最高位优先)两种方法。


以下是 Java 实现的基数排序(LSD 方法)的代码示例和注释。

import java.util.Arrays;
 
public class RadixSort {
 
    // 主排序函数
    public static void radixSort(int[] arr) {
        // 获取数组中的最大值,以确定最大位数
        int max = Arrays.stream(arr).max().getAsInt();
 
        // 从个位开始,对数组进行计数排序
        for (int exp = 1; max / exp > 0; exp *= 10) {
            countingSortByDigit(arr, exp);
        }
    }
 
    // 按位数进行计数排序的辅助函数
    private static void countingSortByDigit(int[] arr, int exp) {
        int n = arr.length;
        int[] output = new int[n]; // 输出数组
        int[] count = new int[10]; // 计数数组,范围为 0-9
 
        // 初始化计数数组为 0
        Arrays.fill(count, 0);
 
        // 统计每个桶中的元素个数
        for (int i = 0; i < n; i++) {
            count[(arr[i] / exp) % 10]++;
        }
 
        // 将计数数组转换为实际位置数组
        for (int i = 1; i < 10; i++) {
            count[i] += count[i - 1];
        }
 
        // 按当前位数将元素放入输出数组中
        for (int i = n - 1; i >= 0; i--) {
            output[count[(arr[i] / exp) % 10] - 1] = arr[i];
            count[(arr[i] / exp) % 10]--;
        }
 
        // 将排序好的元素复制回原数组
        System.arraycopy(output, 0, arr, 0, n);
    }
 
    // 测试函数
    public static void main(String[] args) {
        int[] arr = {170, 45, 75, 90, 802, 24, 2, 66};
        System.out.println("未排序数组: " + Arrays.toString(arr));
        radixSort(arr);
        System.out.println("已排序数组: " + Arrays.toString(arr));
    }
}

上述代码实现了基数排序的 LSD 方法。以下是对代码的详细说明:


主排序函数 radixSort:

radixSort 函数首先通过 Arrays.stream(arr).max().getAsInt() 找到数组中的最大值,用于确定排序的最大位数。

通过 for 循环,从个位开始,每次排序按位数的增加(exp 从 1 到 10、100、1000 ...),调用 countingSortByDigit 函数对数组进行计数排序。


按位数进行计数排序的辅助函数 countingSortByDigit:

countingSortByDigit 函数首先初始化两个数组:output 用于存储排序后的结果,count 用于计数各个桶中的元素个数。

通过 for 循环,统计当前位数(由 exp 确定)的每个桶中的元素个数。

累加计数数组,将计数数组转换为实际位置数组。

通过倒序的 for 循环,按当前位数将元素放入输出数组中。

最后,将排序好的元素复制回原数组。


测试函数 main:

创建一个整数数组并输出未排序的数组。

调用 radixSort 函数对数组进行排序,并输出已排序的数组。

以下是另一种基数排序(MSD 方法)的代码示例和注释。

import java.util.Arrays;
 
public class MSDRadixSort {
 
    // 主排序函数
    public static void radixSort(int[] arr) {
        radixSort(arr, 0, arr.length - 1, getMaxDigit(arr));
    }
 
    // 递归排序函数
    private static void radixSort(int[] arr, int left, int right, int digit) {
        if (left >= right || digit < 0) {
            return;
        }
 
        // 基数桶数组
        int[] buckets = new int[10];
        int[] temp = new int[right - left + 1];
 
        // 统计各个桶的个数
        for (int i = left; i <= right; i++) {
            int bucketIndex = (arr[i] / digit) % 10;
            buckets[bucketIndex]++;
        }
 
        // 转换为各个桶的起始位置
        for (int i = 1; i < 10; i++) {
            buckets[i] += buckets[i - 1];
        }
 
        // 逆序放置元素到临时数组中
        for (int i = right; i >= left; i--) {
            int bucketIndex = (arr[i] / digit) % 10;
            temp[--buckets[bucketIndex]] = arr[i];
        }
 
        // 将临时数组复制回原数组
        System.arraycopy(temp, 0, arr, left, temp.length);
 
        // 对每个桶进行递归排序
        int start = left;
        for (int i = 0; i < 10; i++) {
            int end = (i == 9) ? right : left + buckets[i] - 1;
            radixSort(arr, start, end, digit / 10);
            start = end + 1;
        }
    }
 
    // 获取数组中最大值的位数
    private static int getMaxDigit(int[] arr) {
        int max = Arrays.stream(arr).max().getAsInt();
        int digit = 1;
        while (max / digit >= 10) {
            digit *= 10;
        }
        return digit;
    }
 
    // 测试函数
    public static void main(String[] args) {
        int[] arr = {170, 45, 75, 90, 802, 24, 2, 66};
        System.out.println("未排序数组: " + Arrays.toString(arr));
        radixSort(arr);
        System.out.println("已排序数组: " + Arrays.toString(arr));
    }
}

上述代码实现了基数排序的 MSD 方法。以下是对代码的详细说明:


主排序函数 radixSort:

radixSort 函数首先调用 getMaxDigit 函数获取数组中最大值的位数。

调用递归函数 radixSort,从最大位数开始进行排序。


递归排序函数 radixSort:

递归函数 radixSort 用于对指定范围的数组进行排序,参数包括数组、左右边界和当前处理的位数。

递归终止条件为左右边界相交或当前位数小于 0。

初始化基数桶数组 buckets 和临时数组 temp。

统计当前位数的各个桶的个数,并转换为各个桶的起始位置。

逆序放置元素到临时数组中,并将临时数组复制回原数组。

对每个桶进行递归排序。


获取数组中最大值的位数 getMaxDigit:

getMaxDigit 函数用于获取数组中最大值的位数。

通过 while 循环,不断将最大值除以 10,直到小于 10 为止。


测试函数 main:

创建一个整数数组并输出未排序的数组。

调用 radixSort 函数对数组进行排序,并输出已排序的数组。

通过以上两个基数排序的实现示例,我们可以看到,基数排序的核心思想是利用计数排序对每个位数进行排序,从而达到整体排序的目的。LSD 方法从低位开始排序,而 MSD 方法从高位开始排序,具体选择哪种方法取决于实际应用场景和需求。


相关文章
|
2月前
|
Java
在 Java 中捕获和处理自定义异常的代码示例
本文提供了一个 Java 代码示例,展示了如何捕获和处理自定义异常。通过创建自定义异常类并使用 try-catch 语句,可以更灵活地处理程序中的错误情况。
91 1
|
7天前
|
SQL Java 数据库连接
如何在 Java 代码中使用 JSqlParser 解析复杂的 SQL 语句?
大家好,我是 V 哥。JSqlParser 是一个用于解析 SQL 语句的 Java 库,可将 SQL 解析为 Java 对象树,支持多种 SQL 类型(如 `SELECT`、`INSERT` 等)。它适用于 SQL 分析、修改、生成和验证等场景。通过 Maven 或 Gradle 安装后,可以方便地在 Java 代码中使用。
99 11
|
11天前
|
JSON Java 数据挖掘
利用 Java 代码获取淘宝关键字 API 接口
在数字化商业时代,精准把握市场动态与消费者需求是企业成功的关键。淘宝作为中国最大的电商平台之一,其海量数据中蕴含丰富的商业洞察。本文介绍如何通过Java代码高效、合规地获取淘宝关键字API接口数据,帮助商家优化产品布局、制定营销策略。主要内容包括: 1. **淘宝关键字API的价值**:洞察用户需求、优化产品标题与详情、制定营销策略。 2. **获取API接口的步骤**:注册账号、申请权限、搭建Java开发环境、编写调用代码、解析响应数据。 3. **注意事项**:遵守法律法规与平台规则,处理API调用限制。 通过这些步骤,商家可以在激烈的市场竞争中脱颖而出。
|
2月前
|
Java
在Java中实现接口的具体代码示例
可以根据具体的需求,创建更多的类来实现这个接口,以满足不同形状的计算需求。希望这个示例对你理解在 Java 中如何实现接口有所帮助。
98 38
|
28天前
|
安全 Java 编译器
深入理解Java中synchronized三种使用方式:助您写出线程安全的代码
`synchronized` 是 Java 中的关键字,用于实现线程同步,确保多个线程互斥访问共享资源。它通过内置的监视器锁机制,防止多个线程同时执行被 `synchronized` 修饰的方法或代码块。`synchronized` 可以修饰非静态方法、静态方法和代码块,分别锁定实例对象、类对象或指定的对象。其底层原理基于 JVM 的指令和对象的监视器,JDK 1.6 后引入了偏向锁、轻量级锁等优化措施,提高了性能。
55 3
|
2月前
|
Java
java小工具util系列4:基础工具代码(Msg、PageResult、Response、常量、枚举)
java小工具util系列4:基础工具代码(Msg、PageResult、Response、常量、枚举)
64 24
|
1月前
|
前端开发 Java 测试技术
java日常开发中如何写出优雅的好维护的代码
代码可读性太差,实际是给团队后续开发中埋坑,优化在平时,没有那个团队会说我专门给你一个月来优化之前的代码,所以在日常开发中就要多注意可读性问题,不要写出几天之后自己都看不懂的代码。
67 2
|
1月前
|
Java 编译器 数据库
Java 中的注解(Annotations):代码中的 “元数据” 魔法
Java注解是代码中的“元数据”标签,不直接参与业务逻辑,但在编译或运行时提供重要信息。本文介绍了注解的基础语法、内置注解的应用场景,以及如何自定义注解和结合AOP技术实现方法执行日志记录,展示了注解在提升代码质量、简化开发流程和增强程序功能方面的强大作用。
90 5
|
1月前
|
存储 算法 Java
Java 内存管理与优化:掌控堆与栈,雕琢高效代码
Java内存管理与优化是提升程序性能的关键。掌握堆与栈的运作机制,学习如何有效管理内存资源,雕琢出更加高效的代码,是每个Java开发者必备的技能。
72 5
|
2月前
|
Java API 开发者
Java中的Lambda表达式:简洁代码的利器####
本文探讨了Java中Lambda表达式的概念、用途及其在简化代码和提高开发效率方面的显著作用。通过具体实例,展示了Lambda表达式如何在Java 8及更高版本中替代传统的匿名内部类,使代码更加简洁易读。文章还简要介绍了Lambda表达式的语法和常见用法,帮助开发者更好地理解和应用这一强大的工具。 ####