首页 > 编程语言 >深入了解基数排序:原理、性能分析与 Java 实现

深入了解基数排序:原理、性能分析与 Java 实现

时间:2023-10-13 23:06:16浏览次数:38  
标签:arr Java int 基数排序 位数 数组 原理 排序

基数排序(Radix Sort)是一种非比较性排序算法,它根据元素的每个位上的值来进行排序。基数排序适用于整数或字符串等数据类型的排序。本文将详细介绍基数排序的原理、性能分析及java实现。

深入了解基数排序:原理、性能分析与 Java 实现_数组

基数排序原理

基数排序的基本原理是按照低位先排序,然后收集;再按照高位排序,再收集;以此类推,直到最高位。这样从最低位排序一直到最高位排序完成后,数列就变成一个有序序列。步骤如下:

  1. 从最低位开始,依次进行一次稳定的计数排序(或桶排序),根据当前位的值将元素分配到不同的桶中。
  2. 每一轮排序根据当前位的值进行桶排序,保证稳定性。
  3. 重复以上步骤直到最高位排序完成。

排序图解:

深入了解基数排序:原理、性能分析与 Java 实现_数组_02

Java代码实现

以下是使用 Java 实现基数排序的示例代码:

public class Test {

    public static void main(String[] args) {
        int[] arr = new int[]{122,38,3738,333,63,436,2};
        System.out.println("原始数组:"+ Arrays.toString(arr));
        radixSort(arr);
        System.out.println("排序后的数组:"+ Arrays.toString(arr));
    }

    //基基数排序(Radix Sort)
    public static void radixSort(int[] arr) {
        //数组为空或者只有一个元素是返回
        if(arr == null || arr.length < 2){
            return;
        }
        //获取数组中的最大值
        int maxVal = Arrays.stream(arr).max().getAsInt();
        //计算最大值的位数
        int numDig  = String.valueOf(maxVal).length();
        //数组长度
        int len = arr.length;
        //位数计数值
        int dig = 1;
        //计算每个位数上的数字的被除数
        int dividend = 1;
        // 用于存储每个桶中元素的出现次数
        int[] order = new int[10];
        // 用于存储每个桶中的数据
        int[][] tempArr = new int[10][len];

        //循环每个位数进行桶排序
        while (dig <= numDig){
            //将数组中的数据按i位的数据放入桶中
            for(int i = 0; i < len; i++){
                //计算当前位数的值
                int index =  (arr[i] / dividend ) % 10  ;
                tempArr[index][order[index]] = arr[i];
                //当前位数的桶计数
                order[index]++;
            }
            //给元素中返回数据的下标
            int l = 0;
            //将按当前位数进行过桶排序的数据放回到源数组中
            for (int j = 0; j < order.length; j++){
                //当前桶中有数据才进行操作
                if(order[j] > 0){
                    for(int k = 0; k < order[j];k++){
                        arr[l++] = tempArr[j][k];
                    }
                    //将计数数组的值设置为0,方便下一位计数
                    order[j] = 0;
                }
            }
            System.out.println("第"+dig+"位排序后的数组:"+ Arrays.toString(arr));
            //位数加1,下次进行下一位的排序
            dig++;
            //被除数乘以10,方便计算下一位数的值的计算
            dividend *= 10;
        }

    }

}

输出结果为:

原始数组:[122, 38, 3738, 333, 63, 436, 2]
第1位排序后的数组:[122, 2, 333, 63, 436, 38, 3738]
第2位排序后的数组:[2, 122, 333, 436, 38, 3738, 63]
第3位排序后的数组:[2, 38, 63, 122, 333, 436, 3738]
第4位排序后的数组:[2, 38, 63, 122, 333, 436, 3738]
排序后的数组:[2, 38, 63, 122, 333, 436, 3738]

性能分析

  • 时间复杂度:基数排序的时间复杂度通常为深入了解基数排序:原理、性能分析与 Java 实现_数据_03,其中n是元素数量,d是数字位数,r是基数的大小。通常情况下,基数排序的时间复杂度为线性的,但它依赖于数据位数。如果位数很大,性能可能会受到影响。
  • 空间复杂度:基数排序的空间复杂度取决于计数排序的使用情况。在处理较大范围的数据时,空间复杂度可能会较高。
  • 稳定性:基数排序通常是稳定的。

实用场景

  • 当处理的数据是整数或字符串时,基数排序是一个理想的选择。例如,对于字符串排序,可以按照字符的ASCII码值进行排序。
  • 当数据集的位数相对较小且分布较为均匀时,基数排序可以表现出良好的性能。它不依赖于比较操作,因此在一些特定情况下可以优于基于比较的排序算法。
  • 在内存充足的情况下,基数排序可以高效地处理大量数据,但在位数非常大的情况下可能会受到内存限制的影响。

总结

综上所述,基数排序是一种高效的排序算法,特别适用于处理位数相对较小且分布较为均匀的整数或字符串。但需要注意,对于位数较大的数据集或内存受限的情况,可能需要考虑其他排序算法来满足要求。

标签:arr,Java,int,基数排序,位数,数组,原理,排序
From: https://blog.51cto.com/xiuji/7852087

相关文章

  • elasticsearch通过Java class类的@Setting和@Mapping来定义索引index
    今天就来和大家讲讲如何将es索引中的mapping和setting在索引index和class联系起来,其实在这个问题也困扰我好久了,一直没有解决,在elasticsearch7.x版本的时候貌似好像可以用request在程序中来建立索引,像Stringindex=“{“mapping”:...}”之类的操作,干起来比较复杂,在elasticsear......
  • 工厂方法模式--Java代码实现
    1、画类图2、Java代码实现其中可知,PWFactory、PW类均为接口类;并且,DESFactory、IDEAFactory类均要实现PWFactory接口;DES、IDEA类均要实现PW接口;具体代码如下://PWFactory.javapackageorg.example;publicinterfacePWFactory{publicPWcreateProduce();}//DE......
  • 恒生电子面试(java)
    感觉和学校学的很像,but做不出来具体的语法忘记了,毕竟是两年前学的。。。 第一次面,没事,加油 ......
  • java 警告 源发行版 17 需要目标发行版 17
    java:警告:源发行版17需要目标发行版17打开setting将两个位置都选为8(本人jdk版本)打开ProjectSettings都更改为自己的jdk的版本有的时候上图的Sources部分确实显示的是对的版本,但是后面的Dependencies却变成别的版本了,也一起改了对应......
  • 在JavaScript中如何检查数组是否包含某个值?
    内容来自DOChttps://q.houxu6.top/?s=在JavaScript中如何检查数组是否包含某个值?在JavaScript中,最简洁、高效的方法来检查数组是否包含某个值是什么?这是我所知的唯一方法:functioncontains(a,obj){for(vari=0;i<a.length;i++){if(a[i]===obj)......
  • 在JavaScript中,如何获取时间戳?
    内容来自DOChttps://q.houxu6.top/?s=在JavaScript中,如何获取时间戳?我想要一个单独的数字,代表当前的日期和时间,就像Unix时间戳一样。毫秒级时间戳要获取自Unix纪元以来的毫秒数,调用Date.now:Date.now()或者使用一元运算符+来调用Date.prototype.valueOf:+newDate......
  • Exception in thread "main" java.security.InvalidKeyException: Wrong key size问题
    问题描述在Java里面使用DES加密算法,然后就爆出这个错误:问题解决换用了另外一种加密解密的函数:SecretKeySpec;即将原来的这种:换成了这种:我是觉得使用DES加密算法时,它一直显示key的字节长度不对,就想着换一种表述方式,又看到了别的友友的经验分享,就换成这样试了试(直接放进mai......
  • java语言编码规范
    今天或者说这周突然意识到一个比较重要的问题,就是java语言的编码规范问题,于是整理了一部分的规范格式并且学习:类名要首字母大写,比如 SupplierService,PaymentOrderAction;不要 supplierService,paymentOrderAction.1.4 方法名首字母小写,如 addOrder() 不要 AddOrder()......
  • Android 的ViewBinding实现的原理
    AndroidViewBinding是一种用于替代传统的findViewById和findViewById的视图绑定方法。它允许你以类型安全的方式访问应用布局中的视图元素,而无需手动查找它们。ViewBinding的实现原理如下:布局文件解析:在编译期间,AndroidGradle插件会扫描项目中的布局文件(XML文件),并为每个......
  • 2023.10.13 JavaScript DOM
    文档对象模型获取对象1.根据id属性值获取,返回单个对象varh1=document.getElementById('h1');2.根据标签名获取,返回对象数组vardivs=document.getElementByTagName('div');3.根据name属性值获取,返回对象数组varhobbys=document.getElementByName('hobby');4.根......