Arrays.sort底层排序算法解析

一直以来,我都认为Java内部排序的算法是快速排序。直到有一天,我在面经上看到了有一道这样子的问题,我才发觉,事情远没有我想的那么简单。

在JDK1.8中,自带的底层排序算法会根据传入的目标数组大小的不同采取不同的排序方法。

小数组排序

在底层源码中,对小数组采用快速排序。如果数组的长度小于 QUICKSORT_THRESHOLD = 286 时,使用快速排序。

sql 复制代码
// Use Quicksort on small arrays
if (right - left < QUICKSORT_THRESHOLD) {
    sort(a, left, right, true);
    return;
}

微小数组排序

对于更小的数组,采用插入排序。当数组长度小于 INSERTION_SORT_THRESHOLD = 47 时,采用插入排序。

css 复制代码
// Use insertion sort on tiny arrays
if (length < INSERTION_SORT_THRESHOLD) {
    if (leftmost) {
        // Insertion sort on smallest arrays
        for (int i = left, j = i; i < right; j = ++i) {
            int ai = a[i + 1];
            while (ai < a[j]) {
                a[j + 1] = a[j];
                if (j-- == left) {
                    break;
                }
            }
            a[j + 1] = ai;
        }
        return;
    }
}

大数组排序

对于大数组,首先进行一次数组遍历,标记数组的升序和降序的起始位置,然后根据数组的结构选择归并排序或快速排序。如果升序和降序序列的数量超过 MAX_RUN_COUNT = 67,则采用快速排序。否则,采用归并排序。

css 复制代码
/*
 * Index run[i] is the start of i-th run
 * (ascending or descending sequence).
 */
int[] run = new int[MAX_RUN_COUNT + 1];
int count = 0; 
run[0] = left;

// Check if the array is nearly sorted
for (int k = left; k < right; run[count] = k) {
    // Ascending sequence
    if (a[k] < a[k + 1]) {
        while (++k <= right && a[k - 1] <= a[k]);
    } 
    // Descending sequence
    else if (a[k] > a[k + 1]) {
        while (++k <= right && a[k - 1] >= a[k]);
        // Reverse the descending sequence
        for (int lo = run[count] - 1, hi = k; ++lo < --hi; ) {
            int t = a[lo]; 
            a[lo] = a[hi]; 
            a[hi] = t;
        }
    } 
    // Equal elements
    else {
        for (int m = MAX_RUN_LENGTH; ++k <= right && a[k - 1] == a[k]; ) {
            if (--m == 0) {
                sort(a, left, right, true);
                return;
            }
        }
    }

    // Check if array is highly unstructured
    if (++count == MAX_RUN_COUNT) {
        sort(a, left, right, true);
        return;
    }
}

总的来说,Arrays.sort 在底层使用了快速排序、插入排序以及归并排序等算法,通过根据数组的大小和结构来选择合适的排序策略,以达到更好的性能。

相关推荐
青木川崎17 小时前
linux面试题
java·linux·运维
浮游本尊17 小时前
Java学习第28天 - 大规模系统设计与性能优化
java
上下翻飞的屁17 小时前
jdbcTemplate执行sql后数据库字段没有更新问题解决
java·数据库·sql
爱宇阳17 小时前
Java Spring Boot 项目 Docker 容器化部署教程
java·spring boot·docker
Luo_xguan17 小时前
一、Netty-高并发IO底层原理(5种主要的IO模型)
java·服务器·netty·nio
ACGkaka_17 小时前
SpringBoot 实战(四十)集成 Statemachine
java·spring boot·后端
开开心心就好17 小时前
电脑音质提升:杜比全景声安装详细教程
java·开发语言·前端·数据库·电脑·ruby·1024程序员节
yoi啃码磕了牙17 小时前
Unity—Localization 多语言
java·数据库·mysql
跟着珅聪学java18 小时前
在Java中判断Word文档中是否包含表格并读取表格内容,可以使用Apache POI库教程
java·开发语言·word