Java中利用lambda函数创建复杂条件比较器

以下代码均基于JDK 8

在刷leetcode1的时候遇到这种问题,根据单词词频进行排序,如果词频相同,那么就按照字典排序。因为要用到比较器,而用Lambda表达式写出来的比较器比较简洁,问题如何写出复合条件下的比较器呢?

比如说,现在需要比较Human人这个类。

public class Human {
    private String name; //姓名
    private int age;	//年龄
    // getter setter及构造函数省略
}

它具有年龄age和名字name两个属性,如何满足以下排序:根据年龄升序排序,如果年龄相同,那么根据名字进行字典升序排序。

实现1. 用comparator提供的方法实现

在JDK 8 及其以上版本,我们可以写出利用一级排序comparing和次级排序thenComparing构造如下Lambda表达式

public void sort() {
    
    List<Human> humans = Lists.newArrayList(
      new Human("Sarah", 12), 
      new Human("Sarah", 10), 
      new Human("Zack", 12)
    );

    humans.sort(
      Comparator.comparing(Human::getName).thenComparing(Human::getAge)
    );
}

实现2. 自定义排序方式

当然,我们也可以自己手写量身定制上述过程,

public void sort() {
    List<Human> humans = Lists.newArrayList(
      new Human("Sarah", 12), 
      new Human("Sarah", 10), 
      new Human("Zack", 12)
    );
    
    humans.sort((lhs, rhs) -> {
        if (lhs.getName().equals(rhs.getName())) {
            return Integer.compare(lhs.getAge(), rhs.getAge());
        } else {
            return lhs.getName().compareTo(rhs.getName());
        }
    });
}

但是如果遇到了Human的引用为Null怎么办。我们可以在实现2中添加空值检测即可。而如果我们倒霉地选择了方法1,有什么补救措施呢?

实际上,Comparator类提供了空值处理方法nullsLast(),它会将空值放在后面。与之相反的还有nullsFirst(),它会把空值放在前面。

public void sortWithNull() {
    List<Human> humans = Lists.newArrayList(null, new Human("Jack", 12), null);
    humans.sort(Comparator.nullsLast(Comparator.comparing(Human::getName).thenComparing(Human::getAge)));
}

详细细节请参考这篇文章2


  1. 692. 前K个高频单词 ↩︎

  2. Java 8 – Powerful Comparison with Lambdas ↩︎

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值