JDK8 中Arrays.sort() 排序方法详解

作者:ljj234567 时间:2023-01-17 10:38:30 

一、引言

在刷算法的时候经常需要对数组进行排序,第一反应就是直接使用java.util包下的Arrays.sort()方法直接排序。但在刷算法时会通过时间复杂度空间复杂度对实现的算法进行评价,因此我们需对Arrays.sort()方法有所了解。

本文先行介绍Arrays.sort()中影响排序方式的几个因素。影响因素主要为数组类型数组大小,结合阈值对排序方式进行选择。

二、Arrays.sort()支持类型

Arrays.sort()重载了很多方法,支持多种数据类型的排序。

JDK8 中Arrays.sort() 排序方法详解

三、核心方法DualPivotQuicksort.sort()

进入Arrays.sort()方法的源码,发现内部主要通过DualPivotQuicksort.sort()方法实现排序。该方法通过数组大小、类型结合几个阈值来决定使用哪种排序方式。

public static void sort(int[] a) {
   DualPivotQuicksort.sort(a, 0, a.length - 1, null, 0, 0);
}

DualPivotQuicksort类中的几个常量都是比较关键的阈值,决定了该数组的排序使用哪种方法排序。

/**
    * 长度小于286的数组,优先使用快排而不是归并
    */
   private static final int QUICKSORT_THRESHOLD = 286;
   /**
    * 长度小于47的数组,优先使用插入而不是快排
    */
   private static final int INSERTION_SORT_THRESHOLD = 47;
   /**
    * 如果是byte数组,长度大于29,计数排序优先于插入排序
    */
   private static final int COUNTING_SORT_THRESHOLD_FOR_BYTE = 29;
   /**
    * 如果是char数组,长度大于3200,计数排序优先于快排
    */
   private static final int COUNTING_SORT_THRESHOLD_FOR_SHORT_OR_CHAR = 3200;

1、一般情况的排序方法选择

简单来说,会先计算需要排序的数组长度为n,再根据n的大小及数组元素类型来决定使用什么排序。

根据前两个阈值QUICKSORT_THRESHOLD(286)和INSERTION_SORT_THRESHOLD(47),我们可以看到大多数情况下,排序方法的使用规则是这样的,我们规定需要排序的数组长度为n。

  • n < 47,使用插入排序

  • 47 <= n < 286,使用快速排序

  • n >= 286,使用归并排序

JDK8 中Arrays.sort() 排序方法详解

2、byte、char类型的排序

但是,当数组类型为byte或者char时,会使用到其他两个阈值

数组类型为byte时,查看源码,当数组长度n(right - left) > 29 (COUNTING_SORT_THRESHOLD_FOR_BYTE),使用计数排序,反之,在小数组的情况下使用插入排序

static void sort(byte[] a, int left, int right) {
       // Use counting sort on large arrays
       if (right - left > COUNTING_SORT_THRESHOLD_FOR_BYTE) {
         int[] count = new int[NUM_BYTE_VALUES];
         ... }  else { // Use insertion sort on small arrays
       }
}

数组类型为char时,查看源码实现,当数组长度n(right - left) < 3200 (COUNTING_SORT_THRESHOLD_FOR_SHORT_OR_CHAR ) ,使用计数排序,反之,使用双轴快排

static void sort(char[] a, int left, int right,
                    char[] work, int workBase, int workLen) {
       // Use counting sort on large arrays
       if (right - left > COUNTING_SORT_THRESHOLD_FOR_SHORT_OR_CHAR) {
           int[] count = new int[NUM_CHAR_VALUES];
         ...
       } else { // Use Dual-Pivot Quicksort on small arrays
           doSort(a, left, right, work, workBase, workLen);
       }
}

来源:https://blog.csdn.net/ka3p06/article/details/130515976

标签:JDK8,Arrays.sort(),排序
0
投稿

猜你喜欢

  • Android Zxing生成二维码经典案例分享

    2022-12-22 10:38:27
  • 详解C#如何加密解密RAR文件

    2023-09-19 16:25:10
  • C#基于XNA生成随机颜色的方法

    2021-07-05 08:34:50
  • IDEA社区版下载安装流程详解(小白篇)

    2021-11-13 12:37:18
  • SpringBoot + validation 接口参数校验的思路详解

    2023-10-09 11:59:12
  • 详解SpringBoot+Lucene案例介绍

    2023-03-05 01:15:22
  • Android搜索框通用版

    2022-12-23 09:04:46
  • servlet异步请求的实现

    2023-07-14 17:11:38
  • java.lang.String类的使用

    2021-12-11 03:57:26
  • android获取手机cpu并判断是单核还是多核

    2021-09-28 22:11:24
  • Struts2相关的面试题整理分享

    2022-04-06 08:02:20
  • 利用Flutter实现“孔雀开屏”的动画效果

    2021-11-04 21:24:17
  • Mybatis resultMap标签继承、复用、嵌套方式

    2023-06-27 13:27:43
  • Android去除AlertDialog的按钮栏的分隔线

    2021-06-06 18:53:54
  • Android实现app应用多语言切换功能

    2023-03-30 20:47:13
  • Mybatis配置之<typeAliases>别名配置元素解析

    2023-08-02 03:09:54
  • Java中的匿名对象定义与用法实例分析

    2023-04-11 20:03:05
  • java实现FTP文件上传与文件下载

    2023-08-16 08:28:38
  • 详解Java8中CompletableFuture类的使用

    2022-06-28 17:39:43
  • MyBatis-Plus 自定义sql语句的实现

    2022-12-24 23:16:16
  • asp之家 软件编程 m.aspxhome.com