Java 容器 & 泛型:四、Colletions.sort 和 Arrays.sort 的算法

本文涉及的产品
容器镜像服务 ACR,镜像仓库100个 不限时长
容器服务 Serverless 版 ACK Serverless,952元额度 多规格
容器服务 Serverless 版 ACK Serverless,317元额度 多规格
简介:

本来准备讲 Map集合 ,还是喜欢学到哪里总结吧。最近面试期准备准备,我是一员,成功被阿里在线笔试秒杀回绝。平常心,继续努力。这次带来 Collections 和 Arrays 类中的经典算法剖析。

 

一、Colletions和Arrays

Collentions 此类完全是服务容器的”包装器“。提供了一些操作或者返回容器的静态方法。而Arrays是用来操作数组的各种方法。其中它们的联系在于其中的Sort方法,也就是这次博客的主题。

 

二、插入,快速、归并基本算法

① 插入排序

{a1},{a2,a3,a4,…,an}}

{{a1⑴,a2⑴},{a3⑴,a4⑴ …,an⑴}}

{{a1(n-1),a2(n-1) ,…},{an(n-1)}}

原理及记忆方法:每次处理就是将无序数列的第一个元素与有序数列的元素从后往前逐个进行比较,找出插入位置,将该元素插入到有序数列的合适位置中。这通俗的是找座位思想。Java版实现如下

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
import java.util.Arrays;
 
public class InsertionSort
{
     public static void main(String[] args)
     {
         int[] intA = new int[]{2,1,3,4,6,7,5};
         System.out.println(Arrays.toString(intA));
         insertionSort(intA);
         System.out.println(Arrays.toString(intA));
     }
     
     public static void insertionSort(int[] a)
     {
         int p,right;
         int temp;
         for (p = 0; p < a.length ; p++)
         {
             temp = a [p];
             /**
              * 将a[p]值往左侧有序列比较,插入。
              */
             for ( right = p ; right > 0 && a[right-1] > temp ; right--)
                 a[right] = a[right-1];// 置换
             a[right] = temp;
         }
     }
}

右键,run一下可以看到控制台结果:

?
1
2
[2, 1, 3, 4, 6, 7, 5]
[1, 2, 3, 4, 5, 6, 7]

 

② 快速排序

QQ图片20150414181507

快排是基于分治策略的算法,不是一种稳定的排序算法,也就是说,多个相同的值的相对位置也许会在算法结束时产生变动。 Java版实现如下:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
package javaBasic.algorithm;
 
import java.util.Arrays;
 
public class QuickSort
{
     public static void main(String[] args)
     {
         int[] intA = new int[]{2,1,3,4,6,7,5};
         System.out.println(Arrays.toString(intA));
         //middleSort(intA, 0, intA.length - 1);
         //System.out.println(Arrays.toString(intA));
         sort(intA, 0, intA.length - 1);
         System.out.println(Arrays.toString(intA));
     }
     
     // 快速排序中的一个划分过程
     public static int  middleSort(int a[] , int left , int right)
     {
         int temp = a[left]; // 作为中间轴数
         while( left  < right )
         {
             /**
              * 从右到左,找到第一个比中间轴数小的,移到左端
              */
             while( left < right && a[right] > temp )
                 right--;
             a[left] = a[right];
             
             /**
              * 从左到右,找到第一个比中间轴数大的,移到右端
              */
             while( left < right && a[left] < temp)
                 left++;
             a[right] = a[left];
         }
         
         /**
          * 将中间轴数赋值
          */
         a[left] = temp;
         return left;
     }
     
     // 快速排序
     public static void sort(int[] a , int left, int right)
     {
         if (left < right)
         {
             /**
              * 根据左右索引相同才停止。
              * 不同的话,按着分治思想。
              * 找到中间轴数,一分为二,以此类推。
              */
             int middle = middleSort(a, left, right);
             sort(a, left, middle - 1);
             sort(a, middle + 1, right);
         }
     }
     
}

记忆方法:分治,就是分工。这里演示的是对分。大量经验数据表面,采用两个枢轴来划分成3份的算法更高效,这就是DualPivotQuicksort。这样也是我们后面讲的JDK源码。右键,run一下可以看到控制台和插入排序一样的结果。

③ 归并排序

c8177f3e6709c93d673b9ed49d3df8dcd000[1]

如图,来自百度百科。归并排序也是一种分治思想的算法,之不用快速是对分。归并是一种分解到合并的算法。如下实现方式:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
package javaBasic.algorithm;
 
import java.util.Arrays;
 
public class MergeSort
{
     public static void main(String[] args)
     {
         int[] intA = new int[]{10,4,6,3,8,2,5,7};
         System.out.println(Arrays.toString(intA));
         mergeSort(intA,0,intA.length-1);
         System.out.println(Arrays.toString(intA));
     }
     
     public static void mergeSort(int[] a, int left ,int right)
     {
         if (left < right)
         {
             int middle = (left + right) / 2; // 中间索引
             
             mergeSort(a, left, middle); // 对左侧数组递归
             mergeSort(a, middle+1, right); // 对右侧数组递归
             
             merge(a,left,middle,right); // 归并算法
         }
     }
 
     private static void merge(int[] a, int left, int middle, int right)
     {
         int [] tmpArr = new int[a.length];
         
         int mid = middle+1;
         int tmpArrLeft = left;// 记录左侧数组的索引
         int tmpLeft = left;
         
         /**
          * 从两个数组中取出小的一部分复制
          */
         while (left <= middle && mid <= right)
         {
             if (a[left] <= a[mid])
                 tmpArr[tmpArrLeft++] = a[left++];
             else
                 tmpArr[tmpArrLeft++] = a[mid++];
         }
         
         /**
          * 剩余部分右侧复制
          */
         while (mid <= right)
         {
             tmpArr[tmpArrLeft++] = a[mid++];
         }
         
         /**
          * 剩余部分左侧复制
          */
         while (left <= middle)
         {
             tmpArr[tmpArrLeft++] = a[left++];
         }
         
         /**
          * 分了再合
          */
         while(tmpLeft <= right)
         {
             a[tmpLeft] = tmpArr[tmpLeft++];
         }
     }
     
}

结果和上图一样:

?
1
2
[10, 4, 6, 3, 8, 2, 5, 7]
[2, 3, 4, 5, 6, 7, 8, 10]

 

三、JDK数则

在此谢谢@江南白衣大哥的文章,对我帮助很大。以下引用的:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
5. JDK7/8中排序算法的改进
面试季的同学背一脑袋的插入、归并、冒泡、快排,那,JDK到底看上了哪家的排序算法?
 
Colletions.sort(list) 与 Arrays.sort(T[])
Colletions.sort()实际会将list转为数组,然后调用Arrays.sort(),排完了再转回List。
而Arrays.sort(),对原始类型(int[],double[],char[],byte[]),JDK6里用的是快速排序,对于对象类型(Object[]),JDK6则使用归并排序。为什么要用不同的算法呢?
 
JDK7的进步
到了JDK7,快速排序升级为双基准快排(双基准快排 vs 三路快排);归并排序升级为归并排序的改进版TimSort,一个JDK的自我进化。
 
JDK8的进步
再到了JDK8, 对大集合增加了Arrays.parallelSort()函数,使用fork-Join框架,充分利用多核,对大的集合进行切分然后再归并排序,而在小的连续片段里,依然使用TimSort与DualPivotQuickSort。
 
结论
JDK团队的努力,从一些简单的New Features / Change List 根本看不到,所以没事升级一下JDK还是好的

 

我也查看了关于算法追踪到DualPivotQuicksort类,但是这类却不在JDK API。(抛出个问题:为什么这个类不出现在API里面?)哈哈,一看作者是Java之父参与写的,瞬间有研究的激情。根据白衣大哥说的,快速排序由双基准排序到三路快速排序。这也是在大量经验数据表面,采用两个枢轴来划分成3份的算法更高效。算法的思想也是分治思想。

下面又看到了一段发人自省的注释:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
/**
     * If the length of an array to be sorted is less than this
     * constant, Quicksort is used in preference to merge sort.
     *  当数组长度小于286,为什么快速排序比归并排序好?
     */
    private static final int QUICKSORT_THRESHOLD = 286;
 
    /**
     * If the length of an array to be sorted is less than this
     * constant, insertion sort is used in preference to Quicksort.
     * 当数组长度小于47,为什么插入排序比快速排序好?
     */
    private static final int INSERTION_SORT_THRESHOLD = 47;

 

为什么?第二个问题,欢迎大神解答。

我的理解:第一,建立在大量经验数据结果。第二,根据算法时间复杂度和空间复杂度。至于深入了解需要大神解答。

JDK排序顺序图如下:

绘图1

相关文章
|
2月前
|
Kubernetes Cloud Native Java
云原生之旅:从容器到微服务的演进之路Java 内存管理:垃圾收集器与性能调优
【8月更文挑战第30天】在数字化时代的浪潮中,企业如何乘风破浪?云原生技术提供了一个强有力的桨。本文将带你从容器技术的基石出发,探索微服务架构的奥秘,最终实现在云端自由翱翔的梦想。我们将一起见证代码如何转化为业务的翅膀,让你的应用在云海中高飞。
|
2月前
|
安全 Java 编译器
揭秘JAVA深渊:那些让你头大的最晦涩知识点,从泛型迷思到并发陷阱,你敢挑战吗?
【8月更文挑战第22天】Java中的难点常隐藏在其高级特性中,如泛型与类型擦除、并发编程中的内存可见性及指令重排,以及反射与动态代理等。这些特性虽强大却也晦涩,要求开发者深入理解JVM运作机制及计算机底层细节。例如,泛型在编译时检查类型以增强安全性,但在运行时因类型擦除而丢失类型信息,可能导致类型安全问题。并发编程中,内存可见性和指令重排对同步机制提出更高要求,不当处理会导致数据不一致。反射与动态代理虽提供运行时行为定制能力,但也增加了复杂度和性能开销。掌握这些知识需深厚的技术底蕴和实践经验。
54 2
|
2月前
|
Java Linux Maven
java依赖冲突解决问题之容器加载依赖jar包如何解决
java依赖冲突解决问题之容器加载依赖jar包如何解决
|
17天前
|
Java 编译器 容器
Java——包装类和泛型
包装类是Java中一种特殊类,用于将基本数据类型(如 `int`、`double`、`char` 等)封装成对象。这样做可以利用对象的特性和方法。Java 提供了八种基本数据类型的包装类:`Integer` (`int`)、`Double` (`double`)、`Byte` (`byte`)、`Short` (`short`)、`Long` (`long`)、`Float` (`float`)、`Character` (`char`) 和 `Boolean` (`boolean`)。包装类可以通过 `valueOf()` 方法或自动装箱/拆箱机制创建。
25 9
Java——包装类和泛型
|
20天前
|
安全 Java API
【Java面试题汇总】Java基础篇——String+集合+泛型+IO+异常+反射(2023版)
String常量池、String、StringBuffer、Stringbuilder有什么区别、List与Set的区别、ArrayList和LinkedList的区别、HashMap底层原理、ConcurrentHashMap、HashMap和Hashtable的区别、泛型擦除、ABA问题、IO多路复用、BIO、NIO、O、异常处理机制、反射
【Java面试题汇总】Java基础篇——String+集合+泛型+IO+异常+反射(2023版)
|
17天前
|
存储 安全 搜索推荐
Java中的泛型
【9月更文挑战第15天】在 Java 中,泛型是一种编译时类型检查机制,通过使用类型参数提升代码的安全性和重用性。其主要作用包括类型安全,避免运行时类型转换错误,以及代码重用,允许编写通用逻辑。泛型通过尖括号 `&lt;&gt;` 定义类型参数,并支持上界和下界限定,以及无界和有界通配符。使用泛型需注意类型擦除、无法创建泛型数组及基本数据类型的限制。泛型显著提高了代码的安全性和灵活性。
|
2月前
|
设计模式 缓存 算法
揭秘策略模式:如何用Java设计模式轻松切换算法?
【8月更文挑战第30天】设计模式是解决软件开发中特定问题的可重用方案。其中,策略模式是一种常用的行为型模式,允许在运行时选择算法行为。它通过定义一系列可互换的算法来封装具体的实现,使算法的变化与客户端分离。例如,在电商系统中,可以通过定义 `DiscountStrategy` 接口和多种折扣策略类(如 `FidelityDiscount`、`BulkDiscount` 和 `NoDiscount`),在运行时动态切换不同的折扣逻辑。这样,`ShoppingCart` 类无需关心具体折扣计算细节,只需设置不同的策略即可实现灵活的价格计算,符合开闭原则并提高代码的可维护性和扩展性。
40 2
|
2月前
|
安全 Java Go
Java&Go泛型对比
总的来说,Java和Go在泛型的实现和使用上各有特点,Java的泛型更注重于类型安全和兼容性,而Go的泛型在保持类型安全的同时,提供了更灵活的类型参数和类型集的概念,同时避免了运行时的性能开销。开发者在使用时可以根据自己的需求和语言特性来选择使用哪种语言的泛型特性。
38 7
|
2月前
|
安全 算法 Java
java系列之~~网络通信安全 非对称加密算法的介绍说明
这篇文章介绍了非对称加密算法,包括其定义、加密解密过程、数字签名功能,以及与对称加密算法的比较,并解释了非对称加密在网络安全中的应用,特别是在公钥基础设施和信任网络中的重要性。
|
2月前
|
存储 安全 Java
如何理解java的泛型这个概念
理解java的泛型这个概念
下一篇
无影云桌面