Java> Comparable接口和Comparator接口
java中,如何利用现有数据结构对对象数组Arrays进行排序?
利用Arryas.sort()方法排序,需要结合有两种常用方法:1.使用Comparable接口;2.使用Comparator接口;
2种方法共同点
- 都支持泛型;
- 都是需要实现接口方法compareTo();
- 都需要利用Arrays.sort()对数组进行排序;
2种方法不同点
- Comparable需要修改待排序的对象,也就是要让待排序对象实现Comparable接口方法。而Comparator不影响原来的对象,新建类实现该接口即可;
- 实现Comparable的compareTo(T t)接口只需要传入1个参数,因为另外一个对象就是当前对象(this)。实现Comparator的compareTo(T t1, T t2)需要传入2个参数,因为当前对象并不是待排序对象;
Comparable用法
步骤
- 待排序数组对象实现接口Comparable
; - 实现compareTo方法,将当前对象与形成对象进行比较,较小者将会排在前面(按递增顺序);
- 利用Arrays.sort对数组进行递增排序,本质是用快速排序方法;
- 检查排序结果;
例子
假设需要对一个Employee数组进行排序,按薪水高低排序,薪水较小的排在前,较大排在后.
使用Comparable
public class Employee implements Comparable<Employee> { // 必须是待比较的对象所属类实现Compable接口, 也就是说需要修改待比较 or 排序类
private String name;
private double salary;
public Employee (String name, double salary) {
this.name = name;
this.salary = salary;
}
public int getSalary(){
return salary;
}
public int getName() {
retunr name;
}
@override
public int compareTo(Employee other) { // compareTo 只需要一个参数, 因为this默认指当前待比较对象
return Double.compare(e.getSalary(), other.getSalary());
}
}
静态函数Arrays.sort()是对数组对象进行快速排序,要求对象必须实现了Comparable接口,而且元素之间比较是通过compareTo进行。
当进行x.compareTo(y)时,如果y是x子类则没有问题,而如果y是x父类,则会出现异常。
public class EmployeeTest{
public void main(String[] argv) {
// create data source
Employee[] employees = new Employee[3];
employees[0] = new Employee("henrry", 15000);
employees[1] = new Employee("jok", 20000);
employees[2] = new Employee("martin", 13000);
Arrays.sort(employees); // sort
for (Employee e : employees) { // check result
System.out.println("name = " + e.getName() + ", salary = " + e.getSalary());
}
}
}
运行输出
name = martin, salary = 13000.0
name = henrry, salary = 15000.0
name = jok, salary = 20000.0
Comparator用法
步骤
- 创建待排序数组对象;
- 创建待排序对象的排序对象,实现接口 Comparator
; - 实现compareTo方法,将传入2个同样对象参数进行比较,较小者将会排在前面(按递增顺序);
- 利用Arrays.sort对数组进行递增排序,本质是用快速排序方法;
- 检查排序结果;
例子
还是跟Comparable一样的需求例子,不过实现会有差异。待排序类Employee不再实现Comparable或者Comparator接口。
public class Employee {
private String name;
private double salary;
public Employee(String name, double salary) {
this.name = name;
this.salary = salary;
}
public String getName() {
return name;
}
public double getSalary() {
return salary;
}
}
public class EmployeeComparator implments Comparator<Employee> { // 实现的Comparator的类与待比较类没有直接关系, 不需要修改待比较 or 排序的类
@Override
public int compareTo(Employee e1, Employee e2) { // compareTo 需要传入2个参数, 分别代表要比较的2个对象
return return Double.compare(e1.getSalary() , e2.getSalary());
}
}
public class EmployeeComparatorTest {
public void main(String [] argvs) {
// create data source
Employee[] employees = new Employee[3];
employees[0] = new Employee("henrry", 15000);
employees[1] = new Employee("jok", 20000);
employees[2] = new Employee("martin", 13000);
Arrays.sort(employees, new EmployeeComparator()); // sort
for (Employee e : employees) { // check result
System.out.println("name = " + e.getName() + ", salary = " + e.getSalary());
}
}
}
运行结果(与Comparable的例子一样))
name = martin, salary = 13000.0
name = henrry, salary = 15000.0
name = jok, salary = 20000.0
注意:
x.compareTo(y)会将y转换成x的同类型进行比较,也就是说如果x是y的基类,则没有问题;
如果反过来,y.compareTo(x),x还是y的基类就会出现异常,因为会先将x转换成y的同类型;