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 方法从高位开始排序,具体选择哪种方法取决于实际应用场景和需求。


相关文章
|
12天前
|
Kubernetes jenkins 持续交付
从代码到k8s部署应有尽有系列-java源码之String详解
本文详细介绍了一个基于 `gitlab + jenkins + harbor + k8s` 的自动化部署环境搭建流程。其中,`gitlab` 用于代码托管和 CI,`jenkins` 负责 CD 发布,`harbor` 作为镜像仓库,而 `k8s` 则用于运行服务。文章具体介绍了每项工具的部署步骤,并提供了详细的配置信息和示例代码。此外,还特别指出中间件(如 MySQL、Redis 等)应部署在 K8s 之外,以确保服务稳定性和独立性。通过本文,读者可以学习如何在本地环境中搭建一套完整的自动化部署系统。
38 0
|
5天前
|
存储 Java 开发者
【Java新纪元启航】JDK 22:解锁未命名变量与模式,让代码更简洁,思维更自由!
【9月更文挑战第7天】JDK 22带来的未命名变量与模式匹配的结合,是Java编程语言发展历程中的一个重要里程碑。它不仅简化了代码,提高了开发效率,更重要的是,它激发了我们对Java编程的新思考,让我们有机会以更加自由、更加创造性的方式解决问题。随着Java生态系统的不断演进,我们有理由相信,未来的Java将更加灵活、更加强大,为开发者们提供更加广阔的舞台。让我们携手并进,共同迎接Java新纪元的到来!
29 11
|
2天前
|
并行计算 Java 开发者
探索Java中的Lambda表达式:简化代码,提升效率
Lambda表达式在Java 8中引入,旨在简化集合操作和并行计算。本文将通过浅显易懂的语言,带你了解Lambda表达式的基本概念、语法结构,并通过实例展示如何在Java项目中应用Lambda表达式来优化代码,提高开发效率。我们将一起探讨这一现代编程工具如何改变我们的Java编码方式,并思考它对程序设计哲学的影响。
|
2天前
|
安全 Java 测试技术
掌握Java的并发编程:解锁高效代码的秘密
在Java的世界里,并发编程就像是一场精妙的舞蹈,需要精准的步伐和和谐的节奏。本文将带你走进Java并发的世界,从基础概念到高级技巧,一步步揭示如何编写高效、稳定的并发代码。让我们一起探索线程池的奥秘、同步机制的智慧,以及避免常见陷阱的策略。
|
12天前
|
Java Devops 持续交付
探索Java中的Lambda表达式:简化代码,提升效率DevOps实践:持续集成与部署的自动化之路
【8月更文挑战第30天】本文深入探讨了Java 8中引入的Lambda表达式如何改变了我们编写和管理代码的方式。通过简化代码结构,提高开发效率,Lambda表达式已成为现代Java开发不可或缺的一部分。文章将通过实际例子展示Lambda表达式的强大功能和优雅用法。
|
12天前
|
Java
用JAVA架建List集合为树形结构的代码方法
这段代码定义了一个表示树形结构的 `Node` 类和一个用于构建树形结构的 `TreeController`。`Node` 类包含基本属性如 `id`、`pid`、`name` 和 `type`,以及子节点列表 `children`。`TreeController` 包含初始化节点列表并将其转换为树形结构的方法。通过过滤和分组操作实现树形结构的构建。详情可见:[代码示例链接1](http://www.zidongmutanji.com/zsjx/43551.html),[代码效果参考链接2](https://www.257342.com/sitemap/post.html)。
25 5
|
10天前
|
Java API 开发者
代码小妙招:用Java轻松获取List交集数据
在Java中获取两个 `List`的交集可以通过 `retainAll`方法和Java 8引入的流操作来实现。使用 `retainAll`方法更为直接,但会修改原始 `List`的内容。而使用流则提供了不修改原始 `List`、更为灵活的处理方式。开发者可以根据具体的需求和场景,选择最适合的方法来实现。了解和掌握这些方法,能够帮助开发者在实际开发中更高效地处理集合相关的问题。
10 1
|
12天前
|
Java
java代码和详细的代码应用
代码块分为局部、构造、静态和同步代码块。局部代码块控制变量生命周期,例如 `int a` 只在特定代码块内有效。构造代码块用于创建对象时执行附加功能,避免构造方法中代码重复。静态代码块随类加载执行一次,常用于初始化操作。同步代码块确保多线程环境下方法执行的原子性,通过 `synchronized` 关键字实现。
22 3
|
13天前
|
设计模式 算法 Java
【揭秘】如何巧妙运用Java模板方法模式,让你的代码优雅升级?
【8月更文挑战第30天】模板方法模式是一种行为型设计模式,它定义了算法的骨架并将某些步骤延迟到子类中,使子类能在不改变算法结构的情况下重定义特定步骤。此模式适用于具有共同结构但细节不同的场景,如角色升级系统。通过定义一个抽象类 `Character` 包含模板方法 `levelUp` 和抽象步骤方法,子类如 `Warrior` 和 `Mage` 可以实现具体逻辑。这种方式提供了良好的扩展性,确保算法结构不变,同时保持系统的稳定性和一致性,在数据处理和业务流程管理中广泛应用。
35 2
|
11天前
|
Java
Java中的Lambda表达式:简化代码,提升效率
【8月更文挑战第31天】Lambda表达式在Java 8中引入,旨在使代码更加简洁和易读。本文将探讨Lambda表达式的基本概念、使用场景及如何通过Lambda表达式优化Java代码。我们将通过实际示例来展示Lambda表达式的用法和优势,帮助读者更好地理解和应用这一特性。