先来看一下Comparable的例子,定义实体类Student,实现Comparable,重写compareTo方法:
public class Student implements Comparable<Student> {
private String name;
private Integer age;
private Integer score;
public Student(String name, Integer age, Integer score) {
this.name = name;
this.age = age;
this.score = score;
}
@Override
public int compareTo(Student o) {
return this.getName().compareTo(o.getName());
}
}
测试
public static void main(String[] args) {
Student student1 = new Student("zhangsan", 1, 80);
Student student2 = new Student("lisi", 3, 90);
Student student3 = new Student("wangwu", 2, 100);
List<Student> list = new ArrayList<>();
list.add(student1);
list.add(student2);
list.add(student3);
Collections.sort(list);
list.stream().forEach(n -> System.out.println(n.toString()));
}
结果
Student{name=‘lisi’, age=3, score=90}
Student{name=‘wangwu’, age=2, score=100}
Student{name=‘zhangsan’, age=1, score=80}
从上面的例子我们大致了解了Comparable接口的使用,也就是说同一个类的对象之间如果要进行比较,需要实现Comparable接口,并且实现compareTo方法。这样比较的时候就会按照这个规则来进行比较。
再来看一下Comparator的例子,定义实体类Student
public class Student {
private String name;
private Integer age;
private Integer score;
public Student(String name, Integer age, Integer score) {
this.name = name;
this.age = age;
this.score = score;
}
}
自定义比较器:
class AgeComparator implements Comparator<Student> {
@Override
public int compare(Student o1, Student o2) {
if (o1.getAge() > o2.getAge()) {
return 1;
} else if (o1.getAge() < o2.getAge()) {
return -1;
} else {
return 0;
}
}
}
class NameComparator implements Comparator<Student> {
@Override
public int compare(Student o1, Student o2) {
return o1.getName().compareTo(o2.getName());
}
}
测试
public static void main(String[] args) {
Student student1 = new Student("zhangsan", 1, 80);
Student student2 = new Student("lisi", 3, 90);
Student student3 = new Student("wangwu", 2, 100);
List<Student> list = new ArrayList<>();
list.add(student1);
list.add(student2);
list.add(student3);
// 这时候如果直接 Collections.sort(list) 会由于Student没有默认的自然排序,编译不过。
Collections.sort(list, new AgeComparator());
list.stream().forEach(n -> System.out.println(n.toString()));
System.out.println("\n-------------------");
Collections.sort(list, new NameComparator());
list.stream().forEach(n -> System.out.println(n.toString()));
}
可以看到,我们如果要对实体类的对象进行比较,在不修改原实体类的情况下,可以通过实现多个Comparator来实现多个比较规则。通过Comparator,我们可以自定义比较规则,针对对象的属性或者字段等来进行比较,而Comparable就实现不了,因为它的compareTo方法只能有一种比较规则。
实现Comparator,同样也要实现它的一个方法compare。由于一般情况下我们实现的Comparator只有一个compare方法,所以我们可以对实现类进行一些优化:
①使用匿名类来代替单独的实现类。比如我们可以将 Collections.sort(list, new NameComparator());替换为:
Collections.sort(list, new Comparator<Student>() {
@Override
public int compare(Student o1, Student o2) {
return o1.getName().compareTo(o2.getName());
}
});
②借助JDK1.8的lambda表达式,进一步优化为:
Collections.sort(list, (o1, o2) -> o1.getName().compareTo(o2.getName()));
③借助JDK1.8中Comparator接口中的新的方法comparing,再次优化:
Collections.sort(list, Comparator.comparing(Student::getName));
相同点: 两者都是用来用作对象之间的比较,都可以自定义比较规则;两者都是返回一个描述对象之间关系的int;
不同点: ①实现了Comparable的意思是我可以把自己和另一个对象进行比较;而实现了Comparator的意思是我可以比较其他两个对象;也就是说Comparable是一个可比较的对象可以将自己与另一个对象进行比较;而Comparator是比较两个不同的对象。
② 使用Comparable需要修改原先的实体类,是属于一种自然排序。而Comparator则不用修改原先类。
③即使修改了Comparable实体类,Comparable也仅有一种比较规则。而Comparator可以实现多个,来提供多个比较规则。