Java比较器:Comparable和Comparator

1、是什么

Comparable和Comparator都只是一个普通的接口。他们是Java用来定义如何比较两个对象大小的规范,真正的比较逻辑需要由实现类来重写其中的比较方法。

2、为什么

在Java中,两个数值类型的变量可以使用比较运算符(">","<"等)直接比较两个数值的大小,但是对于引用类型的数据,则不能直接通过比较运算符(">","<"等)来比较两个对象的大小。所以,Java另外提供了两种用来比较两个对象大小的方式:Comparable和Comparator。

3、怎么用

(1)Comparable(自然排序)

java.lang.Comparable 需要子类来实现该接口,并重写当中的compareTo(Object obj)方法,一旦定义完成后就可以在任何地方对该类的对象进行比较排序(可以认为是在对象内部排序)。

直接看Comparable的源码,里面只定义了一个抽象方法 compareTo(Object obj)。

package java.lang;

public interface Comparable<T> {

    int compareTo(T var1);
}

对于任何一个类,只要实现了 java.lang.Comparable 中的compareTo() 方法,该类的对象就可以使用定义的方式进行比较。Java中的包装类和String类都实现了该接口。下面以 java.lang.String 类为例:

@Test
public void test01() {
    List<String> list = Arrays.asList("FF", "AA", "BB", "DD", "TT");
    System.out.println("before:" + list);
    //实现Comparable接口的对象列表(和数组)可以通过 Collections.sort 或 Arrays.sort进行自动排序。
    //实现此接口的对象可以用作有序映射中的键或有序集合中的元素,无需指定比较器。
    Collections.sort(list);
    System.out.println("after:" + list);
}

输出结果:
before:[FF, AA, BB, DD, TT]
after:[AA, BB, DD, FF, TT]

查看String源码可以看到,String类实现了Comparable接口,并且重写了compareTo() 方法,所以可以直接使用Collections.sort()方法进行排序,当两个String对象进行比较时,底层实际是通过从前往后比较每个char字符的Unicode值大小。

public final class String
    implements java.io.Serializable, Comparable<String>, CharSequence {
        
      public int compareTo(String anotherString) {
        int len1 = value.length;
        int len2 = anotherString.value.length;
        int lim = Math.min(len1, len2);
        char v1[] = value;
        char v2[] = anotherString.value;

        int k = 0;
        while (k < lim) {
            char c1 = v1[k];
            char c2 = v2[k];
            if (c1 != c2) {
                return c1 - c2;
            }
            k++;
        }
        return len1 - len2;
    } 
   // ...其他代码省略
}

接着,我们再来自己定义一个Person类来进一步说明:

public class Person implements Comparable<Person> {

    String name;
    int age;

    public Person() {
    }

    public Person(String name, int age) {
        this.name = name;
        this.age = age;
    }

    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }

    /**
     * 要求:先按照age从小到大,age相同时,再按照name从大到小排序
     *
     * @param p 和 this对象进行比较的对象
     * @return
     * 返回1:  this大于p
     * 返回0:  this等于p
     * 返回-1: this小于p
     */
    @Override
    public int compareTo(Person p) {
        if (this.age != p.age) {
            return Integer.compare(this.age, p.age);
        } else {
            //注意这里加 “-”号
            return -this.name.compareTo(p.name);
        }
    }

}

下面进行测试:

@Test
public void test02() {
    List<Person> list = new ArrayList<>();
    list.add(new Person("aa", 20));
    list.add(new Person("cc", 12));
    list.add(new Person("bb", 17));
    list.add(new Person("dd", 18));
    list.add(new Person("ee", 18));

    System.out.println("before:" + list);
    Collections.sort(list);
    System.out.println("after:" + list);
}

输出结果:
before:[Person{name='aa', age=20}, Person{name='cc', age=12}, Person{name='bb', age=17}, Person{name='dd', age=18}, Person{name='ee', age=18}]
after:[Person{name='cc', age=12}, Person{name='bb', age=17}, Person{name='ee', age=18}, Person{name='dd', age=18}, Person{name='aa', age=20}]

但是,实现Comparable的方式也有一些局限性,对于一些特殊场合,如果不想使用原来类中定义的比较方式,而是想使用其他比较方式,又或者想用于比较的对象是第三方库的类,不能进行修改的情况,实现Comparable的方式就不太适合了,这时就可以使用Java提供的第二种排序方式:Comparator。

(2)Comparator(定制排序)

java.util.Comparator 字面意思是比较器,是用来对两个对象进行比较的工具(可以认为是在对象外部排序)。

/**
 * 要求:使用定制排序:先按照name从小到大,name相同时,再按照age从大到小排序
 */
@Test
public void test04() {
    List<Person> list = new ArrayList<>();
    list.add(new Person("aa", 20));
    list.add(new Person("cc", 12));
    list.add(new Person("bb", 17));
    list.add(new Person("ee", 18));
    list.add(new Person("dd", 18));

    System.out.println("before:" + list);
    //因为只使用一次,所以可以使用匿名内部类的方式
    Collections.sort(list, new Comparator<Person>() {
        @Override
        public int compare(Person o1, Person o2) {
            //这里就没有判空了
            if (!o1.name.equals(o2.name)) {
                return o1.name.compareTo(o2.name);
            } else {
                //名字相同就比较年龄,注意这里要加“-”号
                return -Integer.compare(o1.age, o2.age);
            }
        }
    });
    System.out.println("after:" + list);
}

输出结果:
before:[Person{name='aa', age=20}, Person{name='cc', age=12}, Person{name='bb', age=17}, Person{name='ee', age=18}, Person{name='dd', age=18}]
after:[Person{name='aa', age=20}, Person{name='bb', age=17}, Person{name='cc', age=12}, Person{name='dd', age=18}, Person{name='ee', age=18}]

(3)两种方式对比总结:

(1)从名称中可以看出,Comparable以“able”结尾,意思是“可xxx的”,java中也有很多其他类似命令的类,如:Cloneable等。他们的意思基本都是用来定义一种规范或声明,具体的实现细节由实现类来完成。而Comparator是以“or”结尾,一般指的是该类是一个工具型的类。类似的也有:Executor等。

(2)当元素的类型没有实现java.lang.Comparable接口而又不方便修改代码, 或者实现了java.lang.Comparable接口的排序规则不适合当前的操作,那 么可以考虑使用 Comparator 的对象来排序

(3)a. 重写compareTo(Object obj)方法时:

  • 如果当前对象this大于形参obj,则返回正整数;
  • 如果当前对象this等于形参obj,则返回0;
  • 如果当前对象this小于形参obj,则返回负整数。

         b. 重写compare(Object o1,Object o2)方法:

  • 如果o1大于o2,则返回正整数;
  • 如果 o1等于o2,则返回0;
  • 如果 o1小于o2,则返回负整数。

         记忆方法:可以认为:返回数的值 = 前面对象 - 后面对象。

(4)Java中的包装类和String都实现了Comparable接口,重写了compareTo(Object obj)方法,实现了比较两个对象大小的方式。其中:

  • String:按照字符串中每个字符的Unicode编码值进行比较;
  • Character:按照字符的Unicode编码值进行比较;
  • Integer、Long、Float、Double、Byte等数值类型对应的包装类:按照他们对应的数值大小进行比较
  • Boolean:true对应的包装类实例 > false对应的包装类实例;
  • Date、Time等日期时间类:后面的日期大于前面的日期大小(实际比较的是它们对应的时间戳大小)。

(5)实现了Comparable接口的类或者使用调用带Comparator实现类的方式,都可以使用java中的Arrays.sort()方法,或者Collections.sort()方法对多个对象进行排序,Collections.sort() 底层其实也是使用的Arrays.sort(),底层实际就是使用对象实现的compareTo(Object obj)或compare(Object o1, Object o2)方法对数组或集合进行排序。可以参考另一篇文章:https://blog.csdn.net/u011410529/article/details/56668545

(6)由于 null 不是一个类,也不是一个对象,因此在重写 compareTo 方法时应该注意 e.compareTo(null) 的情况,即使 e.equals(null) 返回 false,compareTo 方法也应该主动抛出一个空指针异常 NullPointerException。

(7)Comparable 实现类重写 compareTo 方法时一般要求 e1.compareTo(e2) == 0 的结果要和 e1.equals(e2) 一致。这样将来使用 SortedSet 等根据类的自然排序进行排序的集合容器时可以保证保存的数据的顺序和想象中一致。

 

参考地址:

Java 解惑:Comparable 和 Comparator 的区别:https://zhuanlan.zhihu.com/p/24081048

Arrays.sort和Collections.sort实现原理解析:https://blog.csdn.net/u011410529/article/details/56668545

推荐文章:

剖析JDK8中Arrays.sort底层原理及其排序算法的选择:https://www.jianshu.com/p/d7ba7d919b80

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: "Comparable" 和 "Comparator" 这个词在 Java 编程语言中都是用于排序的概念。 - "Comparable" 是 Java 类库自带的一个接口,它的实现类可以使用 Java 内置的排序方法,例如 Collections.sort()。如果一个类实现了 "Comparable" 接口,就表示这个类支持比较排序。 - "Comparator" 是一个独立的接口,它的实现类可以用于定义自定义的排序方法,例如 Collections.sort(List, Comparator)。如果某个类没有实现 "Comparable" 接口,那么可以通过使用 "Comparator" 来实现排序。 总的来说,如果一个类已经实现了 "Comparable" 接口,那么可以直接使用它的比较方法进行排序。如果没有实现,则需要使用 "Comparator" 来定义自定义的排序方法。 ### 回答2: ComparableComparator是在Java中用于比较对象的种不同方式。 1. Comparable是一个接口,它允许与它相关的类实现自己的比较规则。实现Comparable接口的类必须实现compareTo()方法,该方法用于定义对象之间的自然排序。compareTo()方法返回一个整数值,表示当前对象与参数对象的比较结果。这个值决定了个对象之间的大小关系。 2. Comparator是一个接口,它允许在不修改源代码的情况下定义一个额外的比较规则。与Comparable不同,实现Comparator接口的类可以独立于被比较的类进行比较Comparator接口要求实现compare()方法,该方法用于定义个参数对象之间的比较结果。compare()方法返回一个整数值,表示对象之间的大小关系。 因此,Comparable是被实现在对象自身内部的排序规则,而Comparator是一个独立的外部排序规则。利用Comparable实现的排序规则是类内部默认的排序规则,而Comparator通过传入不同的比较器来实现多种排序规则。 在使用场景上,Comparable常用于对已有的类进行排序,而Comparator通常用于对现有的类进行定制化的排序需求。 ### 回答3: ComparableComparator都是Java中用于排序的接口,它们主要的区别在于使用的方式和对象类型。 Comparable接口是Java中的一个泛型接口,它定义了一个compareTo()方法,用于比较当前对象和另一个对象的大小。实现Comparable接口的类可以直接通过compareTo()方法进行比较和排序,例如,使用Collections.sort()方法对Comparable对象进行排序。 Comparator接口也是Java中的一个泛型接口,它定义了一个compare()方法,用于比较个对象的大小。Comparator接口是一个独立于被比较的对象的比较器,可以用于实现灵活的比较逻辑。比如,如果一个类已经实现了Comparable接口,但我们想要根据不同的条件进行排序,就可以使用Comparator接口来定义不同的比较器Comparator接口可以作为参数传递给排序方法,如Collections.sort(),来实现定制的排序。 简而言之,Comparable是一个类的内部排序方式,实现Comparable接口的类可以通过compareTo()方法进行大小比较和排序。Comparator是一个独立的比较器,可以用于比较任意类型的对象,通过compare()方法来实现不同的排序逻辑。相比之下,Comparator的灵活性更高,可以用于实现各种不同的排序规则和策略,而Comparable只能用于同一种排序逻辑的对象。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值