当我们给一个整型数组或者浮点型之类的数组排序的时候,很简单就可以达到我们排序的目的,无非是排序算法的问题。那么,如果我们现在想根据对象的一个属性值给一个对象数组进行排序呢?
假如我们现在有一个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 tspeed(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;}}}
这样写确实能实现了效果,但是未必性能很好,由于是自己写的,写起来也会比较麻烦,下面的方法会更好
我们先来看看用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 tspeed(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 tspeed(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());}}}
输出结果:
解析:
顾名思义,arrays类的sort()方法是对一个数组进行排序的方法,sort()方法的参数是一个对象数组,也就是要排序的那个数组,但是有些特别的是,这个对象数组中存储的对象的类必须实现了comparable接口。
comparable接口是用来比较大小的,要实现这个接口,我们必须重写接口中的compareto()方法,这个方法用来比较两个对象的大小,因为方法本身并不知道我我们会根据对象的哪个属性来比较两个对象的大小,因此在这个方法中,我们可以定义我们自己的比较规则。
在上述的代码中,我们调用了double类的比较方法来比较两个对象的speed属性的大小,如果调用方法的car对象的speed属性值比方法参数传进来的car对象的speed值大就返回1,相等就返回0,小于就返回-1。当然,我们也可以自己手写这个部分,无非就是if el判断的事。
之所以有两种写法,是因为第一种使用了泛型,第二种使用了对象的强制转换,可以看到,当我们使用泛型的时候,在调用对象的时候就已经确定了使用接口的时候用的是哪一种对象,因此就不用把对象类型强制转换为我们想要的类型了。而第二种我们没有使用泛型,因此就需要把参数传进来的对象强制转换成我们想要的类型然后再进行操作。推荐使用第一种写法。
接下来,我们在main方法中构建了一个car类型的数组,然后直接调用arrays.sort()方法传进数组作为参数就可以对数组进行排序了。
这种方法比较方便,快捷,在性能方面也有不错的表现,本人亲测在排序20万个对象的时候花费差不多200ms左右的时间。
当然,这种方法主要图的是方便,当我们需要排序大量的数据的时候,还是应该结合自己数据的特性来具体选择一种排序方式自己写。
首先先来看一下arrays.sort()使用的例子。
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]+ " "); }
运行结果如下:
arrays.sort()升序: 12 29 34 46 55 57 84 89 98
arrays.sort()降序: 98 89 84 57 55 46 34 29 12
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] + " "); }
运行结果如下:
结合文档以及源代码,我们发现,jdk中的arrays.sort()的实现是通过所谓的双轴快排的算法
下面我们以jdk1.8中arrays对int型数组的排序为例来介绍其中使用的双轴快排:
1.判断数组的长度是否大于286,大于则使用归并排序(merge sort),否则执行2。
// u quicksort on small arrays if (right - left < quicksort_threshold) { sort(a, left, right, true); return; } // merge sort ......
2.判断数组长度是否小于47,小于则直接采用插入排序(inrtion sort),否则执行3。
// u inrtion sort on tiny arrays if (length < inrtion_sort_threshold) { // inrtion sort ...... }
3.用公式length/8+length/64+1近似计算出数组长度的1/7。
// inexpensive approximation of length / 7 int venth = (length >> 3) + (length >> 6) + 1;
4.取5个根据经验得出的等距点。
/* * sort five evenly spaced elements around (and including) the * center element in the range. the elements will be ud for * pivot lection as described below. the choice for spacing * the elements was empirically determined to work well on * a wide variety of inputs. */ int e3 = (left + right) >>> 1; // the midpoint int e2 = e3 - venth; int e1 = e2 - venth; int e4 = e3 + venth; int e5 = e4 + venth;
5.将这5个元素进行插入排序
// sort the elements using inrtion sort if (a[e2] < a[e1]) { int t = a[e2]; a[e2] = a[e1]; a[e1]讪笑是什么意思 = t; } if (a[e3] < a[e哑父2]) { 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; } } } }
6.选取a[e2],a[e4]分别作为pivot1,pivot2。由于步骤5进行了排序,所以必有pivot1 <=pivot2。定义两个指针less和great,less从最左边开始向右遍历,一直找到第一个不小于pivot1的元素,great从右边开始向左遍历,一直找到第一个不大于pivot2的元素。
/* * u the cond and fourth of the five sorted elements as pivots. * the values are inexpensive approximations of the first and * cond terciles of the array. note that pivot1 <= pivot2. */ int pivot1 = a[e2]; int pivot2 = a[e4]; /* * the first and the last elements to be sorted are moved to the * locations formerly occupied by the pivots. when partitioning * is complete, the pivots are swapped back into their final * positions, and excluded from subquent sorting. */ a[e2] = a[left]; a[e4] = a[right]; /* * skip elements, which are less or greater than pivot values. */ while英语6级多少分过 (a[++less] < pivot1); while (a[--great] > pivot2);
7.接着定义指针k从less-1开始向右遍历至great,把小于pivot1的元素移动到less左边,大于pivot2的元素移动到great右边。这里要注意,我们已知great处的元素小于pivot2,但是它于pivot1的大小关系,还需要进行判断,如果比pivot1还小,需要移动到到less左边,否则只需要交换到k处。
/* * partitioning: * * left part center part right part * +--------------------------------------------------------------+ * | < pivot1 | pivot1 <= && <= pivot2 | ? | > pivot2 | * +--------------------------------------------------------------+ * ^ ^ ^ * | | | * less k great * * invariants: * * all in (left, less) < pivot1 * pivot1 <= all in [less, k) <= pivot2 * all in (great, right) > pivot2 * * pointer k is the first index of ?-part. */ 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]; /* * here and below we u "a[i] = b; i++;" instead 热爱祖国的成语* of "a[i++] = b;" due to performance issue. */ a[less] = ak; ++less; } el 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; } el { // pivot1 <= a[great] <= pivot2 a[k] = a[great]; } /* * here and below we u "a[i] = b; i--;" instead * of "a[i--] = b;" due to performance issue. */ a[great] = ak; --great; } }
8.将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;
9.至此,less左边的元素都小于pivot1,great右边的元素都大于pivot2,分别对两部分进行同样的递归排序。
// sort left and right parts recursively, excluding known pivots sort(a, left, less - 2, leftmost); sort(a, great + 2, right, fal);
10.对于中间的部分,如果大于4/7的数组长度,很可能是因为重复元素的存在,所以把less向右移动到第一个不等于pivot1的地方,把great向左移动到第一个不等于pivot2的地方,然后再对less和great之间的部分进行递归排序。
/* * if center part is too large (compris > 4/7 of the array), * swap internal pivot values to ends. */ if (less < e1 && e5 < great) { /* * skip elements, which are equal to pivot values. */ while (a[less] == pivot1) { ++less; } while (a[great] == pivot2) { --great; } } ...... // sort center part recursively sort(a, less, great, fal);
算法步骤
1.对于很小的数组(长度小于47),会使用插入排序。
2.选择两个点p1,p2作为轴心,比如我们可以使用第一个元素和最后一个元素。
3.p1必须比p2要小,否则将这两个元素交换,现在将整个数组分为四部分:
(1)第一部分:比p1小的元素。
(2)第二部分:比p1大但是比p2小的元素。
(3)第三部分:比p2大的元素。
(4)第四部分:尚未比较的部分。
在开始比较前,除了轴点,其余元素几乎都在第四部分,直到比较完之后第四部分没有元素。
4.从第四部分选出一个元素a[k],与两个轴心比较,然后放到第一二三部分中的一个。
5.移动l,k,g指向。
6.重复 4 5 步,直到第四部分没有元素。
7.将p1与第一部分的最后一个元素交换。将p2与第三部分的第一个元素交换。
8.递归的将第一二三部分排序。
**总结:**arrays.sort对升序数组、降序数组和重复数组的排序效率有了很大的提升,这里面有几个重大的优化。
1.对于小数组来说,插入排序效率更高,每次递归到小于47的大小时,用插入排序代替快排,明显提升了性能。
2.双轴快排使用两个pivot,每轮把数组分成3段,在没有明显增加比较次数的情况下巧妙地减少了递归次数。
3.pivot的选择上增加了随机性,却没有带来随机数的开销。
4.对重复数据进行了优化处理,避免了不必要交换和递归。
以上为个人经验,希望能给大家一个参考,也希望大家多多支持www.887551.com。
本文发布于:2023-04-04 04:21:04,感谢您对本站的认可!
本文链接:https://www.wtabcd.cn/fanwen/zuowen/70da40daf2ef06c874deee5dbf0f3a84.html
版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系,我们将在24小时内删除。
本文word下载地址:Java使用Arrays.sort()方法实现给对象排序.doc
本文 PDF 下载地址:Java使用Arrays.sort()方法实现给对象排序.pdf
留言与评论(共有 0 条评论) |