Comparable和Comparator 是什么以及区别

来源:互联网 发布:反转二叉树 js 编辑:程序博客网 时间:2024/06/16 05:35

一、Comparable接口

Comparable可以认为是一个内比较器,实现了Comparable接口的类有一个特点,类的实例与实例直接可以比较,依赖compareTo方法的实现,compareTo方法的返回值就是比较的结果,是int,有三种情况:

1、比较者大于被比较者(也就是compareTo方法里面的对象),那么返回正整数

2、比较者等于被比较者,那么返回0

3、比较者小于被比较者,那么返回负整数

写个很简单的例子:

复制代码
复制代码
public class Domain implements Comparable<Domain>{    private String str;    public Domain(String str)    {        this.str = str;    }    public int compareTo(Domain domain)    {        if (this.str.compareTo(domain.str) > 0)            return 1;        else if (this.str.compareTo(domain.str) == 0)            return 0;        else             return -1;    }        public String getStr()    {        return str;    }}
复制代码
复制代码
复制代码
复制代码
public static void main(String[] args)    {        Domain d1 = new Domain("c");        Domain d2 = new Domain("c");        Domain d3 = new Domain("b");        Domain d4 = new Domain("d");        System.out.println(d1.compareTo(d2));        System.out.println(d1.compareTo(d3));        System.out.println(d1.compareTo(d4));    }
复制代码
复制代码

运行结果为:

01-1

注意一下,前面说实现Comparable接口的类是可以支持和自己比较的,但是其实代码里面Comparable的泛型未必就一定要是Domain,将泛型指定为String或者指定为其他任何任何类型都可以----只要开发者指定了具体的比较算法就行。

 

二、Comparator接口

Comparator可以认为是是一个外比较器,可以去实现Comparator接口得到一个自定义比较器有两种情况可以使用实现Comparator接口的方式:

1、一个对象不支持自己和自己比较(不想实现Comparable接口),但是又想对两个对象进行比较

2、一个对象实现了Comparable接口,但是开发者认为compareTo方法中的比较方式并不是自己想要的那种比较方式

Comparator接口里面有一个compare方法,方法有两个参数T o1和T o2,是泛型的表示方式,分别表示待比较的两个对象,方法返回值也是比较结果和Comparable接口一样是int,有三种情况:

1、o1大于o2,返回正整数

2、o1等于o2,返回0

3、o1小于o3,返回负整数

写个很简单的例子,上面代码的Domain不变(假设这就是第2种场景,我对这个compareTo算法实现不满意,要自己写实现):

复制代码
复制代码
public class DomainComparator implements Comparator<Domain>{    public int compare(Domain domain1, Domain domain2)    {        if (domain1.getStr().compareTo(domain2.getStr()) > 0)            return 1;        else if (domain1.getStr().compareTo(domain2.getStr()) == 0)            return 0;        else             return -1;    }}
复制代码
复制代码
复制代码
复制代码
public static void main(String[] args){    Domain d1 = new Domain("c");    Domain d2 = new Domain("c");    Domain d3 = new Domain("b");    Domain d4 = new Domain("d");    DomainComparator dc = new DomainComparator();    System.out.println(dc.compare(d1, d2));    System.out.println(dc.compare(d1, d3));    System.out.println(dc.compare(d1, d4));}
复制代码
复制代码

看一下运行结果:

01-1

当然因为泛型指定死了,所以实现Comparator接口的实现类只能是两个相同的对象(不能一个Domain、一个String)进行比较了,因此实现Comparator接口的实现类一般都会以"待比较的实体类+Comparator"来命名

 

总结

总结一下,两种比较器Comparable和Comparator,后者相比前者有如下优点:

1、如果实现类没有实现Comparable接口,又想对两个类进行比较(或者实现类实现了Comparable接口,但是对compareTo方法内的比较算法不满意),那么可以实现Comparator接口,自定义一个比较器,写比较算法

2、实现Comparable接口的方式比实现Comparator接口的耦合性 要强一些,如果要修改比较算法,要修改Comparable接口的实现类,而实现Comparator的类是在外部进行比较的,不需要对实现类有任何修 改。从这个角度说,其实有些不太好,尤其在我们将实现类的.class文件打成一个.jar文件提供给开发者使用的时候。实际上实现Comparator 接口的方式后面会写到就是一种典型的策略模式

当然,这不是鼓励用Comparator,意思是开发者还是要在具体场景下选择最合适的那种比较器而已。


参考自:https://www.cnblogs.com/szlbm/p/5504634.html


原创粉丝点击
热门问题 老师的惩罚 人脸识别 我在镇武司摸鱼那些年 重生之率土为王 我在大康的咸鱼生活 盘龙之生命进化 天生仙种 凡人之先天五行 春回大明朝 姑娘不必设防,我是瞎子 新种的月季苗弱怎么办? 月季换盆后浇透水叶子黄了怎么办 肉肉移栽后浇透水怎么办 月季花扦插的没长根发芽了怎么办 君子兰发的小苗怎么办 蔷薇光长枝条不开花怎么办 牡丹发芽又干了怎么办 擦皮炎平后皮肤变黑怎么办 误喝发霉的咖啡渣怎么办 狗吃了速溶咖啡怎么办 咖啡机放豆的地方进水怎么办 干吃咖啡粉上瘾怎么办 去良友花艺住宿怎么办 充气娃娃放了气怎么办 煮杜鹃根没有锅怎么办 淘宝店卖鲜花被买家拒收货怎么办 执业医师电子注册忘记密码怎么办 怀孕吃了油炸的怎么办 百合长得太高怎么办 百合的杆没了怎么办 百合花长得太细怎么办 沙漠玫瑰的花苞打不开怎么办 鲜切花 较小的花苞怎么办 大棚玫瑰苗水大涝的不长怎么办 鲜花买回来蔫了怎么办 喝玫瑰醋上火了怎么办 插在花泥上的花怎么办 插的花蔫了怎么办 紫睡莲的茎软了怎么办 家养的荷花烂叶怎么办 家养的荷花叶老是枯萎怎么办 新买的绣球蔫了怎么办 绣球花被太阳晒阉了怎么办 羊肉香精放多了怎么办 被飞机防腐剂弄到皮肤怎么办 狗吃了脱氧保鲜剂呕吐怎么办 小孩误吃试纸了保鲜剂怎么办 狗狗把保鲜剂吃了怎么办 小孩吃了防潮珠怎么办 狗吃了防潮剂怎么办 洋桔梗有点烂根怎么办