转自http://www.cnblogs.com/skywang12345/p/3324788.html

本文,先介绍Comparable 和Comparator两个接口,以及它们的差异;接着,通过示例,对它们的使用方法进行说明。


Comparable 简介

Comparable 是排序接口。

若一个类实现了Comparable接口,就意味着“该类支持排序”。  即然实现Comparable接口的类支持排序,假设现在存在“实现Comparable接口的类的对象的List列表(或数组)”,则该List列表(或数组)可以通过 Collections.sort(或 Arrays.sort)进行排序。

此外,“实现Comparable接口的类的对象”可以用作“有序映射(如TreeMap)”中的键或“有序集合(TreeSet)”中的元素,而不需要指定比较器。

Comparable 定义

Comparable 接口仅仅只包括一个函数,它的定义如下:

package java.lang;
import java.util.*;public interface Comparable<T> {public int compareTo(T o);
}

说明:

假设我们通过 x.compareTo(y) 来“比较x和y的大小”。若返回“负数”,意味着“x比y小”;返回“零”,意味着“x等于y”;返回“正数”,意味着“x大于y”。


Comparator 简介

Comparator 是比较器接口。

我们若需要控制某个类的次序,而该类本身不支持排序(即没有实现Comparable接口);那么,我们可以建立一个“该类的比较器”来进行排序。这个“比较器”只需要实现Comparator接口即可。

也就是说,我们可以通过“实现Comparator类来新建一个比较器”,然后通过该比较器对类进行排序。

Comparator 定义

Comparator 接口仅仅只包括两个个函数,它的定义如下:

package java.util;public interface Comparator<T> {int compare(T o1, T o2);boolean equals(Object obj);
}

说明:

(01) 若一个类要实现Comparator接口:它一定要实现compareTo(T o1, T o2) 函数,但可以不实现 equals(Object obj) 函数。

为什么可以不实现 equals(Object obj) 函数呢? 因为任何类,默认都是已经实现了equals(Object obj)的。 Java中的一切类都是继承于java.lang.Object,在Object.java中实现了equals(Object obj)函数;所以,其它所有的类也相当于都实现了该函数。

(02) int compare(T o1, T o2) 是“比较o1和o2的大小”。返回“负数”,意味着“o1比o2小”;返回“零”,意味着“o1等于o2”;返回“正数”,意味着“o1大于o2”。


Comparator 和 Comparable 比较

Comparable是排序接口;若一个类实现了Comparable接口,就意味着“该类支持排序”。
而Comparator是比较器;我们若需要控制某个类的次序,可以建立一个“该类的比较器”来进行排序。

我们不难发现:Comparable相当于“内部比较器”,而Comparator相当于“外部比较器”。

我们通过一个测试程序来对这两个接口进行说明。源码如下:

import java.util.*;
import java.lang.Comparable;/*** @desc "Comparator"和“Comparable”的比较程序。*   (01) "Comparable"*   它是一个排序接口,只包含一个函数compareTo()。*   一个类实现了Comparable接口,就意味着“该类本身支持排序”,它可以直接通过Arrays.sort() 或 Collections.sort()进行排序。*   (02) "Comparator"*   它是一个比较器接口,包括两个函数:compare() 和 equals()。*   一个类实现了Comparator接口,那么它就是一个“比较器”。其它的类,可以根据该比较器去排序。**   综上所述:Comparable是内部比较器,而Comparator是外部比较器。*   一个类本身实现了Comparable比较器,就意味着它本身支持排序;若它本身没实现Comparable,也可以通过外部比较器Comparator进行排序。*/
public class CompareComparatorAndComparableTest{public static void main(String[] args) {// 新建ArrayList(动态数组)ArrayList<Person> list = new ArrayList<Person>();// 添加对象到ArrayList中list.add(new Person("ccc", 20));list.add(new Person("AAA", 30));list.add(new Person("bbb", 10));list.add(new Person("ddd", 40));// 打印list的原始序列System.out.printf("Original  sort, list:%s\n", list);// 对list进行排序// 这里会根据“Person实现的Comparable<String>接口”进行排序,即会根据“name”进行排序Collections.sort(list);System.out.printf("Name      sort, list:%s\n", list);// 通过“比较器(AscAgeComparator)”,对list进行排序// AscAgeComparator的排序方式是:根据“age”的升序排序Collections.sort(list, new AscAgeComparator());System.out.printf("Asc(age)  sort, list:%s\n", list);// 通过“比较器(DescAgeComparator)”,对list进行排序// DescAgeComparator的排序方式是:根据“age”的降序排序Collections.sort(list, new DescAgeComparator());System.out.printf("Desc(age) sort, list:%s\n", list);// 判断两个person是否相等testEquals();}/*** @desc 测试两个Person比较是否相等。*   由于Person实现了equals()函数:若两person的age、name都相等,则认为这两个person相等。*   所以,这里的p1和p2相等。**   TODO:若去掉Person中的equals()函数,则p1不等于p2*/private static void testEquals() {Person p1 = new Person("eee", 100);Person p2 = new Person("eee", 100);if (p1.equals(p2)) {System.out.printf("%s EQUAL %s\n", p1, p2);} else {System.out.printf("%s NOT EQUAL %s\n", p1, p2);}}/*** @desc Person类。*       Person实现了Comparable接口,这意味着Person本身支持排序*/private static class Person implements Comparable<Person>{int age;String name;public Person(String name, int age) {this.name = name;this.age = age;}public String getName() {return name;}public int getAge() {return age;}public String toString() {return name + " - " +age;}/*** 比较两个Person是否相等:若它们的name和age都相等,则认为它们相等*/boolean equals(Person person) {if (this.age == person.age && this.name == person.name)return true;return false;}/*** @desc 实现 “Comparable<String>” 的接口,即重写compareTo<T t>函数。*  这里是通过“person的名字”进行比较的*/@Overridepublic int compareTo(Person person) {return name.compareTo(person.name);//return this.name - person.name;}}/*** @desc AscAgeComparator比较器*       它是“Person的age的升序比较器”*/private static class AscAgeComparator implements Comparator<Person> {@Override public int compare(Person p1, Person p2) {return p1.getAge() - p2.getAge();}}/*** @desc DescAgeComparator比较器*       它是“Person的age的升序比较器”*/private static class DescAgeComparator implements Comparator<Person> {@Override public int compare(Person p1, Person p2) {return p2.getAge() - p1.getAge();}}}

下面对这个程序进行说明。

a) Person类定义。如下:

private static class Person implements Comparable<Person>{int age;String name;.../** * @desc 实现 “Comparable<String>” 的接口,即重写compareTo<T t>函数。*  这里是通过“person的名字”进行比较的*/@Overridepublic int compareTo(Person person) {return name.compareTo(person.name);//return this.name - person.name;}
}

说明:

(01) Person类代表一个人,Persong类中有两个属性:age(年纪) 和 name“人名”。
(02) Person类实现了Comparable接口,因此它能被排序

b) 在main()中,我们创建了Person的List数组(list)。如下:

// 新建ArrayList(动态数组)
ArrayList<Person> list = new ArrayList<Person>();
// 添加对象到ArrayList中
list.add(new Person("ccc", 20));
list.add(new Person("AAA", 30));
list.add(new Person("bbb", 10));
list.add(new Person("ddd", 40));

c) 接着,我们打印出list的全部元素。如下:

// 打印list的原始序列
System.out.printf("Original sort, list:%s\n", list);

d) 然后,我们通过Collections的sort()函数,对list进行排序。

由于Person实现了Comparable接口,因此通过sort()排序时,会根据Person支持的排序方式,即 compareTo(Person person) 所定义的规则进行排序。如下:

// 对list进行排序
// 这里会根据“Person实现的Comparable<String>接口”进行排序,即会根据“name”进行排序
Collections.sort(list);
System.out.printf("Name sort, list:%s\n", list);

e) 对比Comparable和Comparator

我们定义了两个比较器 AscAgeComparator 和 DescAgeComparator,来分别对Person进行 升序 和 降低 排序。

e.1) AscAgeComparator比较器

它是将Person按照age进行升序排序。代码如下:

/*** @desc AscAgeComparator比较器*       它是“Person的age的升序比较器”*/
private static class AscAgeComparator implements Comparator<Person> {@Overridepublic int compare(Person p1, Person p2) {return p1.getAge() - p2.getAge();}
}

e.2) DescAgeComparator比较器

它是将Person按照age进行降序排序。代码如下:

/*** @desc DescAgeComparator比较器*       它是“Person的age的升序比较器”*/
private static class DescAgeComparator implements Comparator<Person> {@Overridepublic int compare(Person p1, Person p2) {return p2.getAge() - p1.getAge();}
}

f) 运行结果

运行程序,输出如下:

Original  sort, list:[ccc - 20, AAA - 30, bbb - 10, ddd - 40]
Name      sort, list:[AAA - 30, bbb - 10, ccc - 20, ddd - 40]
Asc(age)  sort, list:[bbb - 10, ccc - 20, AAA - 30, ddd - 40]
Desc(age) sort, list:[ddd - 40, AAA - 30, ccc - 20, bbb - 10]
eee - 100 EQUAL eee - 100

  

转载于:https://www.cnblogs.com/danbing/p/5096008.html

Java 中 Comparable 和 Comparator 比较(转)相关推荐

  1. Java中Comparable和Comparator接口区别分析

    本文要来详细分析一下Java中Comparable和Comparator接口的区别,两者都有比较的功能,那么究竟有什么区别呢,感兴趣的Java开发者继续看下去吧. Comparable 简介 Comp ...

  2. 简单介绍Java中Comparable和Comparator

    转载自 简单介绍Java中Comparable和Comparator Comparable 和 Comparator是Java核心API提供的两个接口,从它们的名字中,我们大致可以猜到它们用来做对象之 ...

  3. java comparator相等_详解Java中Comparable和Comparator接口的区别

    详解Java中Comparable和Comparator接口的区别 发布于 2020-7-20| 复制链接 摘记: 详解Java中Comparable和Comparator接口的区别本文要来详细分析一 ...

  4. Java中Comparable和Comparator实现对象比较

    Java中Comparable和Comparator实现对象比较 目录 一. Comparator 二. Comparable 三.比较 当需要排序的集合或数组不是单纯的数字型时,通常可以使用Comp ...

  5. JAVA语言-比较器Comparator(java中Comparable和Comparator的区别)

    文章目录 一.什么是Comparator 二.Java compare方法和compareTo方法 三.java中Comparable和Comparator的区别 Comparator的例子 三.de ...

  6. Java 中 Comparable 和 Comparator 比较

    为什么80%的码农都做不了架构师?>>>    本文,先介绍Comparable 和Comparator两个接口,以及它们的差异:接着,通过示例,对它们的使用方法进行说明. Comp ...

  7. Java中Comparable和Comparator的区别

    前言 最近复习遇到了这个问题,在此进行一个详细的记录,分享给大家. 两个接口的区别 包区别 Comparable接口是在java.lang下. Comparator接口是在java.util下. 使用 ...

  8. java compareto方法怎么排序的_深入理解Java中Comparable和Comparator排序

    本文有牛旦教育原创,头条首发,转载注明来源. 如何为需要的排序算法选择正确的接口?通过本文的分析讲解,我们会找到答案参考答案. 程序员经常需要将数据库中的元素排序为集合.数组或映射.在Java中,我们 ...

  9. Java中Comparable和Comparator区别小结

    阅读目录 一.Comparable简介 二.Comparator简介 三.Comparable和Comparator区别比较 回到顶部 一.Comparable简介 Comparable是排序接口.若 ...

  10. Java中Comparable与Comparator的区别

    一.概述 Comparable和Comparator都是用来实现集合中元素的比较.排序的. Comparable是在集合内部定义的方法实现的排序,位于java.util下. Comparator是在集 ...

最新文章

  1. 主成分分析 PCA算法
  2. Java的各种打包方式
  3. 【拆解】华为Mate Xs ,胶水用的出乎意外的少!
  4. 最优化课堂笔记08——非线性规划中的一些其他方法(考试你懂得)
  5. const型数据小结
  6. pythonsql注入步骤_防止SQL注入解决方案
  7. 关于Tomcat的部署
  8. 普开数据第15届全国高校师资班(青海西宁)
  9. 【笔记】LR录制方式和常用函数
  10. 计算机软件系统 七年级,系统软件作用大七年级信息技术教案(5页)-原创力文档...
  11. 5328笔记 Advanced ML Chapter3-Hypothesis Complexity and Generalisation
  12. 程序员女友在京东被领导“潜规则”,竟然不回绝:表明非单身会影响绩效
  13. 聚焦医疗数字化,华为医疗物联网更懂智慧医疗
  14. 修改参数(parameter)
  15. SM2算法与原生RSA、ECDSA算法的比较
  16. 天源财富:突破“极限”!我科学家发现迄今最高能量光子
  17. C305例会-电脑攒机
  18. 解决ubuntu18.04版本点击右键无法创建空白文档的问题
  19. ubuntu 离线下载安装包 及安装
  20. 数据输入、转换、展示和存储 - 网络统计学类函数(1)

热门文章

  1. 借助Intent实现Android工程中Activity之间Java对象的传递——实现Parcelable接口
  2. readResolve()原理
  3. VS中使用码云gitee建立源代码管理
  4. SQL Server查询sql去掉小数位后末尾的0
  5. js嵌套函数内外层分别使用this关键字困局解
  6. MinIO之C#上传文件等各项操作
  7. 数据库JDBCUtil 工具类 增加连接池操作
  8. 阿里代码规范pdf_看完阿里的代码规范,立马学会代码分层,再也不会被同事怼...
  9. pandas之Dataframe 菜鸟教程
  10. mysql concat算法_MySQL CONCAT()用法及代码示例