返回顶部
首页 > 资讯 > 精选 >Java如何使用Arrays.sort()方法实现给对象排序
  • 402
分享到

Java如何使用Arrays.sort()方法实现给对象排序

2023-06-22 04:06:23 402人浏览 安东尼
摘要

这篇文章主要介绍了Java如何使用Arrays.sort()方法实现给对象排序,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。使用Arrays.sort()方法给对象排序当我们

这篇文章主要介绍了Java如何使用Arrays.sort()方法实现给对象排序,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。

使用Arrays.sort()方法给对象排序

当我们给一个整型数组或者浮点型之类的数组排序的时候,很简单就可以达到我们排序的目的,无非是排序算法的问题。那么,如果我们现在想根据对象的一个属性值给一个对象数组进行排序呢?

假如我们现在有一个Car类型,Car类中有一个double型的speed属性用来描述车辆的速度,现在我们想根据车速来对一个Car数组中的车辆进行排序,怎么做呢?

public class Car{private double speed;//车辆的速度属性public Car(double speed) {this.speed = speed;}public double getSpeed() {return speed;}public void setSpeed(double speed) {this.speed = speed;}}

麻烦的方法

我们可以通过比较对象的属性值,根据比较结果,对对象数组进行排序,例如,假如我们使用的是冒泡排序的话,就需要写成下面这样:

for(int i=0;i<a.length-1;i++)//a为一个存储了很多Car对象的数组{for(int j=0;j<a.length-i-1;j++){if(a[j].getSpeed()>a[j+1].getSpeed())//交换操作{Car temp=a[j];a[j]=a[j+1];a[j+1]=temp;}}}

这样写确实能实现了效果,但是未必性能很好,由于是自己写的,写起来也会比较麻烦,下面的方法会更好

Arrays.sort()方法

我们先来看看用Array.sort()方法实现对车辆排序的代码:

其中,Car这个类有两种写法:

第一种写法:

public class Car implements Comparable<Car>{private double speed;public Car(double speed) {this.speed = speed;}public double getSpeed() {return speed;}public void setSpeed(double speed) {this.speed = speed;}@Overridepublic int compareTo(Car newCar) {return Double.compare(this.getSpeed(),newCar.getSpeed());}}

第二种写法

public class Car implements Comparable{private double speed;public Car(double speed) {this.speed = speed;}public double getSpeed() {return speed;}public void setSpeed(double speed) {this.speed = speed;}@Overridepublic int compareTo(Object newCar) {return Double.compare(this.getSpeed(),((Car)newCar).getSpeed());}}

main方法:

public class Test8 {public static void main(String[] args) {Car[] carList = new Car[3];carList[0] = new Car(30);carList[1] = new Car(10);carList[2] = new Car(55);Arrays.sort(carList);for(Car c:carList){System.out.println(c.getSpeed());}}}

输出结果:


Java如何使用Arrays.sort()方法实现给对象排序

解析:

顾名思义,Arrays类的sort()方法是对一个数组进行排序的方法,sort()方法的参数是一个对象数组,也就是要排序的那个数组,但是有些特别的是,这个对象数组中存储的对象的类必须实现了Comparable接口。

Comparable接口是用来比较大小的,要实现这个接口,我们必须重写接口中的CompareTo()方法,这个方法用来比较两个对象的大小,因为方法本身并不知道我我们会根据对象的哪个属性来比较两个对象的大小,因此在这个方法中,我们可以定义我们自己的比较规则。

在上述的代码中,我们调用了Double类的比较方法来比较两个对象的speed属性的大小,如果调用方法的Car对象的speed属性值比方法参数传进来的Car对象的speed值大就返回1,相等就返回0,小于就返回-1。当然,我们也可以自己手写这个部分,无非就是if else判断的事。

之所以有两种写法,是因为第一种使用了泛型,第二种使用了对象的强制转换,可以看到,当我们使用泛型的时候,在调用对象的时候就已经确定了使用接口的时候用的是哪一种对象,因此就不用把对象类型强制转换为我们想要的类型了。而第二种我们没有使用泛型,因此就需要把参数传进来的对象强制转换成我们想要的类型然后再进行操作。推荐使用第一种写法。

接下来,我们在main方法中构建了一个Car类型的数组,然后直接调用Arrays.sort()方法传进数组作为参数就可以对数组进行排序了。

这种方法比较方便,快捷,在性能方面也有不错的表现,本人亲测在排序20万个对象的时候花费差不多200ms左右的时间。

当然,这种方法主要图的是方便,当我们需要排序大量的数据的时候,还是应该结合自己数据的特性来具体选择一种排序方式自己写。

浅谈Arrays.sort()原理

首先先来看一下Arrays.sort()使用的例子。

例子1

Arrays.sort(int[] a)
 //注意一定要用Integer对象类        Integer[] a1 = {34, 57, 46, 89, 98, 12, 55, 84, 29};        Integer[] a2 = {34, 57, 46, 89, 98, 12, 55, 84, 29};        //增序,Arrays.sort()默认升序        Arrays.sort(a1);        System.out.println("Arrays.sort()升序:");        for (int i = 0; i < a1.length; i++) {            System.out.print(a1[i] + " ");        }        //降序,可用Comparator()匿名内部类        Arrays.sort(a2, new Comparator<Integer>() {            @Override            public int compare(Integer o1, Integer o2) {                return o2.compareTo(o1);            }        });        System.out.println("\nArrays.sort()降序:");        for (int i = 0; i < a2.length; i++) {            System.out.print(a2[i]+ " ");        }

基础知识点

  • 若是基本类型,需要转化为对应的对象类型(如:int转化为Integer)Arrays.sort()可以排序基本对象类型,但是不可以使用基本数据类型。

  • Arrays.sort()默认的是升序排序,降序排序可采用Collection.sort()匿名内部类。

  • 数组与list一样,需要遍历出来。

运行结果如下:

Arrays.sort()升序: 12 29 34 46 55 57 84 89 98
Arrays.sort()降序: 98 89 84 57 55 46 34 29 12

例子2

Arrays.sort(int[] a, int fromIndex, int toIndex)
//注意一定要用Integer对象类Integer[] a1 = {34, 57, 46, 89, 98, 12, 55, 84, 29}; //对数组中的第四位到第7位(不包含第七位)(左闭右开原则)进行排序 Arrays.sort(a1,3,6); System.out.println("Arrays.sort()升序:"); for (int i = 0; i < a1.length; i++) {     System.out.print(a1[i] + " "); }

运行结果如下:

Java如何使用Arrays.sort()方法实现给对象排序

结合文档以及源代码,我们发现,jdk中的Arrays.sort()的实现是通过所谓的双轴快排的算法

双轴快排

  • 快速排序使用的是分治思想,将原问题分成若干个子问题进行递归解决。选择一个元素作为轴(pivot),通过一趟排序将要排序的数据分割成独立的两部分,其中一部分的所有数据都比轴元素小,另外一部分的所有数据都比轴元素大,然后再按此方法对这两部分数据分别进行快速排序,整个排序过程可以递归进行,以此达到整个数据变成有序序列。

  • 双轴快排(DualPivotQuicksort),顾名思义有两个轴元素pivot1,pivot2,且pivot ≤ pivot2,将序列分成三段:x < pivot1、pivot1 ≤ x ≤ pivot2、x >pivot2,然后分别对三段进行递归。这个算法通常会比传统的快排效率更高,也因此被作为Arrays.java中给基本类型的数据排序的具体实现。

下面我们以JDK1.8中Arrays对int型数组的排序为例来介绍其中使用的双轴快排:

判断数组的长度是否大于286,大于则使用归并排序(merge sort),否则执行2。

 // Use Quicksort on small arrays    if (right - left < QUICKSORT_THRESHOLD) {            sort(a, left, right, true);            return;    }    // Merge sort    ......


2.判断数组长度是否小于47,小于则直接采用插入排序(insertion sort),否则执行3。

 // Use insertion sort on tiny arrays    if (length < INSERTION_SORT_THRESHOLD) {    // Insertion sort    ......    }

用公式length/8+length/64+1近似计算出数组长度的1/7。

 // Inexpensive approximation of length / 7    int seventh = (length >> 3) + (length >> 6) + 1;

取5个根据经验得出的等距点。

     int e3 = (left + right) >>> 1; // The midpoint    int e2 = e3 - seventh;    int e1 = e2 - seventh;    int e4 = e3 + seventh;    int e5 = e4 + seventh;

将这5个元素进行插入排序

// Sort these elements using insertion sort    if (a[e2] < a[e1]) { int t = a[e2]; a[e2] = a[e1]; a[e1] = t; }    if (a[e3] < a[e2]) { int t = a[e3]; a[e3] = a[e2]; a[e2] = t;    if (t < a[e1]) { a[e2] = a[e1]; a[e1] = t; }    }    if (a[e4] < a[e3]) { int t = a[e4]; a[e4] = a[e3]; a[e3] = t;        if (t < a[e2]) { a[e3] = a[e2]; a[e2] = t;            if (t < a[e1]) { a[e2] = a[e1]; a[e1] = t; }        }    }    if (a[e5] < a[e4]) { int t = a[e5]; a[e5] = a[e4]; a[e4] = t;        if (t < a[e3]) { a[e4] = a[e3]; a[e3] = t;            if (t < a[e2]) { a[e3] = a[e2]; a[e2] = t;                if (t < a[e1]) { a[e2] = a[e1]; a[e1] = t; }            }        }    }

选取a[e2],a[e4]分别作为pivot1,pivot2。由于步骤5进行了排序,所以必有pivot1 <=pivot2。定义两个指针less和great,less从最左边开始向右遍历,一直找到第一个不小于pivot1的元素,great从右边开始向左遍历,一直找到第一个不大于pivot2的元素。

         int pivot1 = a[e2];        int pivot2 = a[e4];                a[e2] = a[left];        a[e4] = a[right];                while (a[++less] < pivot1);        while (a[--great] > pivot2);

接着定义指针k从less-1开始向右遍历至great,把小于pivot1的元素移动到less左边,大于pivot2的元素移动到great右边。这里要注意,我们已知great处的元素小于pivot2,但是它于pivot1的大小关系,还需要进行判断,如果比pivot1还小,需要移动到到less左边,否则只需要交换到k处。

        outer:        for (int k = less - 1; ++k <= great; ) {            int ak = a[k];            if (ak < pivot1) { // Move a[k] to left part                a[k] = a[less];                                a[less] = ak;                ++less;            } else if (ak > pivot2) { // Move a[k] to right part                while (a[great] > pivot2) {                    if (great-- == k) {                        break outer;                    }                }                if (a[great] < pivot1) { // a[great] <= pivot2                    a[k] = a[less];                    a[less] = a[great];                    ++less;                } else { // pivot1 <= a[great] <= pivot2                    a[k] = a[great];                }                                a[great] = ak;                --great;            }        }

将less-1处的元素移动到队头,great+1处的元素移动到队尾,并把pivot1和pivot2分别放到less-1和great+1处。

// Swap pivots into their final positions        a[left]  = a[less  - 1]; a[less  - 1] = pivot1;        a[right] = a[great + 1]; a[great + 1] = pivot2;

至此,less左边的元素都小于pivot1,great右边的元素都大于pivot2,分别对两部分进行同样的递归排序。

// Sort left and right parts recursively, excluding known pivots        sort(a, left, less - 2, leftmost);        sort(a, great + 2, right, false);

对于中间的部分,如果大于4/7的数组长度,很可能是因为重复元素的存在,所以把less向右移动到第一个不等于pivot1的地方,把great向左移动到第一个不等于pivot2的地方,然后再对less和great之间的部分进行递归排序。

        if (less < e1 && e5 < great) {                        while (a[less] == pivot1) {                ++less;            }            while (a[great] == pivot2) {                --great;            }        }        ......        // Sort center part recursively        sort(a, less, great, false);

另外参考了其他博文,算法思路如下

算法步骤

对于很小的数组(长度小于47),会使用插入排序。

选择两个点P1,P2作为轴心,比如我们可以使用第一个元素和最后一个元素。

P1必须比P2要小,否则将这两个元素交换,现在将整个数组分为四部分:

(1)第一部分:比P1小的元素。

(2)第二部分:比P1大但是比P2小的元素。

(3)第三部分:比P2大的元素。

(4)第四部分:尚未比较的部分。

在开始比较前,除了轴点,其余元素几乎都在第四部分,直到比较完之后第四部分没有元素。

从第四部分选出一个元素a[K],与两个轴心比较,然后放到第一二三部分中的一个。

移动L,K,G指向。

重复 4 5 步,直到第四部分没有元素。

将P1与第一部分的最后一个元素交换。将P2与第三部分的第一个元素交换。

递归的将第一二三部分排序。

**总结:**Arrays.sort对升序数组、降序数组和重复数组的排序效率有了很大的提升,这里面有几个重大的优化。

对于小数组来说,插入排序效率更高,每次递归到小于47的大小时,用插入排序代替快排,明显提升了性能。

双轴快排使用两个pivot,每轮把数组分成3段,在没有明显增加比较次数的情况下巧妙地减少了递归次数。

pivot的选择上增加了随机性,却没有带来随机数的开销。

对重复数据进行了优化处理,避免了不必要交换和递归。

感谢你能够认真阅读完这篇文章,希望小编分享的“Java如何使用Arrays.sort()方法实现给对象排序”这篇文章对大家有帮助,同时也希望大家多多支持编程网,关注编程网精选频道,更多相关知识等着你来学习!

--结束END--

本文标题: Java如何使用Arrays.sort()方法实现给对象排序

本文链接: https://lsjlt.com/news/302371.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • Java如何使用Arrays.sort()方法实现给对象排序
    这篇文章主要介绍了Java如何使用Arrays.sort()方法实现给对象排序,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。使用Arrays.sort()方法给对象排序当我们...
    99+
    2023-06-22
  • Java使用Arrays.sort()方法实现给对象排序
    目录使用Arrays.sort()方法给对象排序麻烦的方法Arrays.sort()方法浅谈Arrays.sort()原理例子1基础知识点例子2双轴快排另外参考了其他博文,算法思路如...
    99+
    2024-04-02
  • Java的Arrays.sort()方法排序算法实例分析
      暂时网上看过很多JDK8中Arrays.sort的底层原理,有些说是插入排序,有些说是归并排序,也有说大于域值用计数排序法,否则就使用插入排序。。。其实不全对。让我们分析个究竟:...
    99+
    2024-04-02
  • golag如何使用sort.slice包实现对象list排序
    这篇文章将为大家详细讲解有关golag如何使用sort.slice包实现对象list排序,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。1.sort.Sort介绍使用sort.Slice进行排序,因为sli...
    99+
    2023-06-29
  • golang 使用sort.slice包实现对象list排序
    目录1.sort.Sort介绍1.1分析内置sort包1.2分析sort.go2.使用方法2.1基础类型排序2.2对象排序(单一字段)2.3对象排序(多字段)3.sort.Slice...
    99+
    2024-04-02
  • java中实现对类的对象进行排序
    我们需要对类按照类中的某一个属性(或者多个属性)来对类的对象进行排序,有两种方法可以实现,一种方法是类实现Comparable接口,然后调用Collections.sort(List)方法进行排序,另一种方法是类不实现Comparable接...
    99+
    2022-03-08
    java教程 java 实现 对象 排序
  • Java排序算法之堆排序如何实现
    这篇文章主要介绍了Java排序算法之堆排序如何实现,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。二叉堆是完全二叉树或者是近似完全二叉树。二叉堆满足二个特性︰1.父结点的键值总...
    99+
    2023-06-21
  • vue给数组中对象排序sort函数的用法
    目录vue给数组中对象排序 sort函数vue小技巧:简单排序和对象排序对于数组里面全是number对于一个对象 有多种类型vue给数组中对象排序 sort函数 开发穿梭框的时候,需...
    99+
    2024-04-02
  • 对切片进行排序:利用排序方法实现切片排序
    目前编程网上已经有很多关于Golang的文章了,自己在初次阅读这些文章中,也见识到了很多学习思路;那么本文《对切片进行排序:利用排序方法实现切片排序》,也希望能帮助到大家,如果阅读完后真的对你学习G...
    99+
    2024-04-04
  • Java排序算法之计数排序如何实现
    这篇文章主要为大家展示了“Java排序算法之计数排序如何实现”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“Java排序算法之计数排序如何实现”这篇文章吧。计数排序是非比较的排序算法,用辅助数组对...
    99+
    2023-06-21
  • 利用java如何实现一个快速排序方法
    本篇文章给大家分享的是有关利用java如何实现一个快速排序方法,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。java 算法之快速排序实现代码摘要: 常用算法之一的快速排序算法的...
    99+
    2023-05-31
    java ava 快速排序
  • 利用java如何实现一个冒泡排序方法
    利用java如何实现一个冒泡排序方法?相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。冒泡排序:基本思想:在要排序的一组数中,对当前还未排好序的范围内的全部数,自上而下对相邻的两个...
    99+
    2023-05-31
    java 冒泡排序 ava
  • Java如何使用Collections.sort对中文进行排序方式
    本篇文章为大家展示了Java如何使用Collections.sort对中文进行排序方式,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。使用Collections.sort对中文进行排序使用collec...
    99+
    2023-06-25
  • java如何实现快速排序算法
    这篇文章将为大家详细讲解有关java如何实现快速排序算法,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。快速排序算法使用的分治法策略来把一个序列分为两个子序列来实现排序的思路:1.从数列中挑出一个元素,称为...
    99+
    2023-06-02
  • Java如何实现HashMap排序
    这篇文章主要介绍了Java如何实现HashMap排序的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇Java如何实现HashMap排序文章都会有所收获,下面我们一起来看看吧。排序已有数据按key排序使用strea...
    99+
    2023-06-30
  • Java如何实现ArrayList排序
    这篇文章主要介绍“Java如何实现ArrayList排序”,在日常操作中,相信很多人在Java如何实现ArrayList排序问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Java如何实现ArrayList排序...
    99+
    2023-06-30
  • Vue中如何给Window对象添加方法
    目录给Window对象添加方法为window对象添加事件处理程序给Window对象添加方法 大家都知道vue中所有元素都是作用于Vue实例的,可是我使用DCloud的Wap2App打...
    99+
    2024-04-02
  • Java使用Collections.sort()排序的方法
    Java中Collections.sort()的使用 在日常开发中,很多时候都需要对一些数据进行排序的操作。然而那些数据一般都是放在一个集合中如:Map ,Set ,List 等集...
    99+
    2024-04-02
  • 在Java中如何给List进行排序?这7种方法都可以轻松实现!
    你好,这里是Java架构历程,每天一个Java知识,愿我们每个Java开发者早日成为Java架构师,加油! 在Java中我们经常需要对 ArrayList、LinkedList 或 CopyOnWri...
    99+
    2023-09-16
    java list python
  • Java对象流如何实现序列化的类
    小编给大家分享一下Java对象流如何实现序列化的类,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!Java有哪些集合类Java中的集合主要分为四类:1、List列表...
    99+
    2023-06-14
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作