使用背景
- Java中的对象,正常情况下,只能进行比较:== 或 != 。不能使用 > 或 < 的
- 但是在开发场景中,我们需要对多个对象进行排序,言外之意,就需要比较对象的大小。
- 如何实现?使用两个接口中的任何一个:Comparable 或 Comparator
自然排序:java.lang.Comparable
- 实现 Comparable 的类必须实现 compareTo(Object obj) 方法,两个对象即通过 compareTo(Object obj) 方法的返回值来比较大小。
- 如果当前对象this大于形参对象obj,则返回正整数
- 如果当前对象this小于形参对象obj,则返回负整数
- 如果当前对象this等于形参对象obj,则返回零。
- 实现Comparable接口的对象列表(和数组)可以通过 Collections.sort 或 Arrays.sort进行自动排序。实现此接口的对象可以用作有序映射中的键或有序集合中的元素,无需指定比较器。
- 对于类 C 的每一个e1和e2来说,当且仅当 e1.compareTo(e2) == 0 与 e1.equals(e2) 具有相同的 boolean 值时,类 C 的自然排序才叫做与 equals 一致。建议(虽然不是必需的)最好使自然排序与 equals 一致。
- Comparable 的典型实现:(默认都是从小到大排列的)
String:按照字符串中字符的Unicode值进行比较
Character:按照字符的Unicode值来进行比较
数值类型对应的包装类以及BigInteger、BigDecimal:按照它们对应的数值大小进行比较
public static void main(String[] args) {
Person[] people = new Person[5];
people[0] = new Person("Tom", 15);
people[1] = new Person("Bob", 32);
people[2] = new Person("Tom", 56);
people[3] = new Person("Gall", 23);
people[4] = new Person("Silly", 66);
Arrays.sort(people);
System.out.println(Arrays.toString(people));
}
}
class Person implements Comparable{
String name;
int age;
public Person(String name, int age) {
this.name = name;
this.age = age;
}
@Override
public int compareTo(Object o) {
if(o instanceof Person){
Person p = (Person)o;
if(this.name.equals(p.name)){
return Integer.compare(this.age,p.age);
}else{
return this.name.compareTo(p.name);
}
}
throw new RuntimeException("类型不匹配");
}
@Override
public String toString() {
return "Person{" +
"name='" + name + '\'' +
", age=" + age +
'}';
}
定制排序java.util.Comparator
- 当元素的类型没有实现java.lang.Comparable接口而又不方便修改代码, 或者实现了java.lang.Comparable接口的排序规则不适合当前的操作,那么可以考虑使用 Comparator类的对象来排序,强行对多个对象进行整体排序的比较。
- 重写compare(Object o1,Object o2)方法,比较o1和o2的大小:
- 如果方法返回正整数,则表示o1大于o2;
- 如果返回0,表示相等;
- 返回负整数,表示 o1小于o2。
- 可以将 Comparator的对象传递给 sort 方法(如 Collections.sort 或 Arrays.sort), 从而允许在排序顺序上实现精确控制。
Arrays.sort(goods,com);
Collections.sort(coll,com);
new TreeSet(com);
//指明比较大小的方式:照名称从低到高排序,再照年龄从高到低排序
Person[] p = new Person[5];
p[0] = new Person("Dom", 15);
p[1] = new Person("Bob", 32);
p[2] = new Person("Dom", 56);
p[3] = new Person("Ali", 23);
p[4] = new Person("Silly", 66);
Arrays.sort(p, new Comparator() {
@Override
public int compare(Object o1, Object o2) {
if(o1 instanceof Person && o2 instanceof Person){
Person p1 = (Person) o1;
Person p2 = (Person) o2;
if(p1.name.equals(p2.name)){
return -Integer.compare(p1.age,p2.age);
}else {
return p1.name.compareTo(p2.name);
}
}
throw new RuntimeException("类型不匹配");
}
});
System.out.println(Arrays.toString(p));
}
两种排序方式对比
Comparable接口的方式一旦一定,保证Comparable接口实现类的对象在任何位置都可以比较大小。
Comparator接口属于临时性的比较。