Java 中的 Comparator 和 Comparable

1. 前言

上一文简单对 Java 中的Comparator 接口进行了说明,今天我们来看看另一个和它非常类似的接口 java.lang.Comparable

2. Comparable

Comparable 接口同样只有一个抽象方法 int compareTo(T o),其规则和Comparatorint compare(T o1, T o2)类似。虽然它也可以看作一个函数式接口,但是 Java 8 中并没有将它标记为函数式接口。说明设计者并不希望开发者将它作为函数式接口来使用。不然就偏离了设计意图,就像下面输入一个字符串返回该字符串的长度。

// 符合语法但是不符合设计意图的操作
Comparable<String> comparable = String::length;

通常情况下Comparable 希望被作为对象的一个特性来表达该对象的实例之间是相互比较的。比如电影有按照年份进行比较的特性。

class Movie implements Comparable<Movie> {
    private double rating;
    private String name;
    private int year;

    // Used to sort movies by year
    public int compareTo(Movie m){
        return this.year - m.year;
    }
}

Comparable 通常用于自然排序,也就是元素本身是可比较的。

3.Comparator vs Comparable

ComparatorComparable 很相似,但是它们也是有一些不一样的地方的,主要表现在:

  • 视角不同,Comparable 通常是对象自带的比较属性而 Comparator 通常是作为“第三方”来比较。

  • 通常 Comparable需要被对象实现来作为特性使用,而Comparator 更像是策略。

  • 一个位于 java.lang 包下 ,一个在 java.util 下,这从侧面也证明了第一条。

4. 总结

总而言之,如果对象的排序需要基于自然顺序(它本身是可比较的),则使用 Comparable,而如果需要根据业务来对不同属性进行排序,请使用 Comparator

往期推荐:

Java 集合排序规则接口 Comparator

Spring MVC 函数式编程进阶

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
JavaComparatorComparable都是用于对象排序的接口,它们的区别在于: 1. Comparable是在对象内部实现的接口,实现了Comparable接口的类可以通过实现compareTo()方法来指定对象之间的自然排序规则。例如,如果一个类实现了Comparable接口,那么它的对象可以直接通过Collections.sort()或Arrays.sort()方法进行排序。 2. Comparator是在对象外部实现的接口,它可以用于对不支持自然排序的类进行排序。Comparator接口定义了一个compare()方法,用于比较两个对象的大小关系。在排序时,可以通过传入一个Comparator对象来指定排序规则。 举个例子,假设我们有一个Circle类,它有radius属性,我们可以通过实现Comparable接口来指定Circle对象之间的自然排序规则,如下所示: ```java public class Circle implements Comparable<Circle> { private double radius; public Circle(double radius) { this.radius = radius; } public double getRadius() { return radius; } @Override public int compareTo(Circle o) { if (this.radius < o.radius) { return -1; } else if (this.radius > o.radius) { return 1; } else { return 0; } } } ``` 在上面的例子,我们通过实现Comparable接口来指定Circle对象之间的自然排序规则,即按照半径从小到大排序。现在我们可以直接使用Collections.sort()方法对Circle对象进行排序,如下所示: ```java List<Circle> circles = new ArrayList<>(); circles.add(new Circle(3)); circles.add(new Circle(1)); circles.add(new Circle(2)); Collections.sort(circles); System.out.println(circles); // 输出:[Circle(radius=1.0), Circle(radius=2.0), Circle(radius=3.0)] ``` 如果我们想要按照半径从大到小排序,可以通过实现Comparator接口来指定排序规则,如下所示: ```java public class CircleComparator implements Comparator<Circle> { @Override public int compare(Circle o1, Circle o2) { if (o1.getRadius() < o2.getRadius()) { return 1; } else if (o1.getRadius() > o2.getRadius()) { return -1; } else { return 0; } } } ``` 在上面的例子,我们通过实现Comparator接口来指定Circle对象之间的排序规则,即按照半径从大到小排序。现在我们可以通过传入一个CircleComparator对象来对Circle对象进行排序,如下所示: ```java List<Circle> circles = new ArrayList<>(); circles.add(new Circle(3)); circles.add(new Circle(1)); circles.add(new Circle(2)); Collections.sort(circles, new CircleComparator()); System.out.println(circles); // 输出:[Circle(radius=3.0), Circle(radius=2.0), Circle(radius=1.0)] ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

码农小胖哥

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值