1、引言
在学习数组时,我们知道可以通过导包来使用Arrays类的sort方法对数组元素进行升序排序:
可那仅限于数据之间可直接进行比较的元素,当数据元素为某个对象时,对象有多个属性,又该如何去比较并进行排序呢?
当我们直接使用排序方法时,发生了报错,这是为什么呢?
让我们带着疑问,继续阅读下去,一起解开谜团。
2、Comparable接口
当我们去运行时发现,idea给出了类型转换异常,并说明Student类没有转换为java.lang.Comparable类型。
当我们点进去源代码中的报错位置:
我们发现每一个元素都被强转为了Comparable接口类型,并调用重写的compareTo方法进行比较,而我们没有实现Comparable接口,更没有重写compareTo方法,所以出现了类型转换异常。
为什么要实现Comparable接口呢?
其实就是因为我们当前数组类型中的元素为自定类型的对象,对象的属性有name也有age,编译器不知道根据哪个进行比较,也就是说没有具备可比较功能,那我们就需要实现Comparable接口,根据自己的意愿来重写compareTo方法(用哪个属性来比较),使其具备可比较功能。
(返回正数则表明当前对象(this)中的值更大,返回负数则表明o对象中的值更大)
注:
这里< >中的内容为要进行比较的类(泛型的知识,知道填什么即可)。
在这里,我重写的compareTo方法是根据年龄来比较的,我们再次来运行代码:
我们发现,数组便根据年龄的大小进行了排序。
当然也可以通过姓名来进行排序:
在这里大家可能会有疑问:为啥name能直接调用comparaTo方法来进行比较?
这是因为,name是String类型,而String类已经实现了Comparable接口并重写了compareTo方法,所以name可以直接调用compareTo方法来进行字符串之间的比较:
2.1 利用重写的compareTo方法模拟实现mySort排序方法
到这里,我们已经理解了Comparable接口和compareTo方法,那我们可以用我们自己的思想来完成对自定义类的排序,这样可以帮助我们理解方法的底层逻辑,更深的理解compareTo方法,知道它是如何使用的:
这样,我们便完成了对数组的排序(根据年龄):
3、Comparator接口
理解了Comparable接口,不知道大家有没有发现一个巨大的缺陷,就是当我们实现了Comparable接口来进行对象之间的比较时,比较方式就已经固定下来了,不够灵活,例如:在上述的代码中,我们实现了comparaTo方法后,只能根据age进行比较,如果我们要更换比较方式(更换为通过name进行比较),那就需要改变comparaTo方法的实现,这是非常死板不灵活的。
如何解决这个问题呢?
那就需要介绍另一种接口:Comparator接口
我们需要重新定义一个类来实现Comparator接口,重写compare方法:
我们可以定义多个类来完成对多个属性的比较:
注意:
1. o1是要比较的新元素,无序序列中的元素(o1进行插入,插入后使序列有序)
2. o2是被比较的旧元素,有序序列中的元素
这时,我们只需要将要比较属性相应类的对象传入Arrays的sort方法中即可(大家可能对这一传参形式陌生,不过不要紧):
这样,我们想要根据哪个属性进行比较,我们就可以传入相应类的对象即可,非常灵活。
compare方法在使用sort方法进行排序的底层原理:
同样,compare方法也不止应用于多组数据的排序:
总而言之:我们想根据哪个属性进行比较或排序,我们只需要使用相应的方法或传入相应参数即可,大大增加了代码的灵活性!!!
ps:
总结:以上两种接口均可完成对自定义类型之间的比较,Comparable接口主要用于固定的比较,而使用Comparator接口来比较更加灵活。
OK~本次博客到这里就结束了,
感谢大家的阅读~欢迎大家在评论区交流问题~
如果博客出现错误可以提在评论区~
创作不易,请大家多多支持~