原文地址:http://leihuang.org/2014/11/16/Comparable-Vs-Comparator/
Comparable和Comparator都是用来实现集合中元素的比较、排序的,只是Comparable 是在集合内部定义的方法实现的排序,Comparator 是在集合外部实现的排序,所以,如想实现排序,就需要在集合外定义 Comparator 接口的方法或在集合内实现 Comparable 接口的方法。
Comparator位于包java.util下,而Comparable位于包 java.lang下.
Comparable 是一个对象本身就已经支持自比较所需要实现的接口(如 String、Integer 自己就可以完成比较大小操作,已经实现了Comparable接口)
自定义的类要在加入list容器中后能够排序,可以实现Comparable接口,在用Collections类的sort方法排序时,如果不指定Comparator,那么就以自然顺序排序,如API所说:
Sorts the specified list into ascending order, according to the natural ordering of its elements. All elements in the list must implement the Comparable interface
这里的自然顺序就是实现Comparable接口设定的排序方式。
而 Comparator 是一个专用的比较器,当这个对象不支持自比较或者自比较函数不能满足你的要求时,你可以写一个比较器来完成两个对象之间大小的比较。
可以说一个是自已完成比较,一个是外部程序实现比较的差别而已。
用 Comparator 是策略模式(strategy design pattern),就是不改变对象自身,而用一个策略对象(strategy object)来改变它的行为。
Comparable interface
例如我们定义一个类Country,我们希望他的对象有大小关系,且依据countryId来排序的,所以我们让Country实现Comparable接口,使其具备大小关系的能力,代码如下:
class Country implements Comparable{
private int countryId ;
@Override
public int compareTo(Object o) {
Country country = (Country)o ;
return this.getCountryId()>country.getCountryId()?1:this.getCountryId()==
country.getCountryId()?0:-1 ;
}
}
此时Country就具备了比较的能力了,如同Integer,String类一样,此时Country对象的集合就可以通过Collections.sort(),或者Arrays.sort(),来进行排序了。
下面我们来看看Comparator使如何实现的。
Comparator interface
就像我们之前说的一样,Comparator其实就好像一个工具,它不是使得Country具有排序性,而是他不改变Country类本身,而是单独实现一个排序工具,来提供给Country用,下面我们就来实现这个排序工具类。
class Country{
private int id ;
public Country(int id){
this.id = id ;
}
public int getId() {
return id;
}
public void setId(int id) {
this.id = id;
}
}
class CountryByIdComparator implements Comparator<Country>{
@Override
public int compare(Country o1, Country o2) {
return o1.getId()>o2.getId()?1:o1.getId()==o2.getId()?0:-1 ;
}
}
大家注意到没,单单看Country类,其实并没有跟比较有任何关系,但是它却能通过实现Comparator借口的工具类ContryByIdComparator来实现自动排序。此时调用的Collections.sort()就需要两个参数了,一个是集合,一个是实现Comparator接口的工具类。
我们也可以利用匿名内部类来实现,而不需要单独定义ContryByIdComparator这样一个工具类,代码如下:
Collections.sort(cityList,new Comparator<City>(){
@Override
public int compare(City o1, City o2) {
return o1.getId()>o2.getId()?1:o1.getId()==o2.getId()?0:-1 ;
}
}) ;
Comparator 是策略模式(strategy design pattern),就是不改变对象自身,而用一个策略对象(strategy object)来改变它的行为。
下面这张图就是它们两的区别:
Refference
2014-11-16 17:27:25
Brave,Happy,Thanksgiving !