经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » Java相关 » Java » 查看文章
Java Comparator.comparing比较导致空指针异常的解决
来源:jb51  时间:2021/7/12 11:32:07  对本文有异议

Java Comparator.comparing比较导致空指针异常

  1. Comparator.comparing(Department::getOrder)

原因:

  1. public static <T, U extends Comparable<? super U>> Comparator<T> comparing(
  2. Function<? super T, ? extends U> keyExtractor)
  3. {
  4. Objects.requireNonNull(keyExtractor);
  5. return (Comparator<T> & Serializable)
  6. (c1, c2) -> keyExtractor.apply(c1).compareTo(keyExtractor.apply(c2));
  7. }

如果keyExtractor.apply(c1),那么keyExtractor.apply(c1).compareTo(XX)将报空指针异常

替代方案

  1. Comparator.comparing(Department::getOrder, Comparator.nullsFirst(Comparator.naturalOrder()))

替代方案好处:

  1. public static <T, U> Comparator<T> comparing(
  2. Function<? super T, ? extends U> keyExtractor,
  3. Comparator<? super U> keyComparator)
  4. {
  5. Objects.requireNonNull(keyExtractor);
  6. Objects.requireNonNull(keyComparator);
  7. return (Comparator<T> & Serializable)
  8. (c1, c2) -> keyComparator.compare(keyExtractor.apply(c1),
  9. keyExtractor.apply(c2));
  10. }

会先取出keyExtractor.apply(c1)和keyExtractor.apply(c2),放入比较器进行比较

而Comparator.nullsFirst作为比较器,会创建一个Comparators.NullComparator比较器

  1. public static <T> Comparator<T> nullsFirst(Comparator<? super T> comparator) {
  2. return new Comparators.NullComparator<>(true, comparator);
  3. }

Comparators.NullComparator比较器的compare接口实现中先进行空值判断处理,不为空的再进行代入比较器比较

  1. /**
  2. * Null-friendly comparators
  3. */
  4. final static class NullComparator<T> implements Comparator<T>, Serializable {
  5. private static final long serialVersionUID = -7569533591570686392L;
  6. private final boolean nullFirst;
  7. // if null, non-null Ts are considered equal
  8. private final Comparator<T> real;
  9. @SuppressWarnings("unchecked")
  10. NullComparator(boolean nullFirst, Comparator<? super T> real) {
  11. this.nullFirst = nullFirst;
  12. this.real = (Comparator<T>) real;
  13. }
  14. @Override
  15. public int compare(T a, T b) {
  16. if (a == null) {
  17. return (b == null) ? 0 : (nullFirst ? -1 : 1);
  18. } else if (b == null) {
  19. return nullFirst ? 1: -1;
  20. } else {
  21. return (real == null) ? 0 : real.compare(a, b);
  22. }
  23. }
  24. @Override
  25. public Comparator<T> thenComparing(Comparator<? super T> other) {
  26. Objects.requireNonNull(other);
  27. return new NullComparator<>(nullFirst, real == null ? other : real.thenComparing(other));
  28. }
  29. @Override
  30. public Comparator<T> reversed() {
  31. return new NullComparator<>(!nullFirst, real == null ? null : real.reversed());
  32. }
  33. }

Comparator中comparing方法的学习

例子:

我们需要根据对象中的name字段进行不规则排序

排序规则为(PPD > 政府 > 合作)

  1. public class Obj {
  2. private String name;
  3. private BigDecimal price;
  4. ......
  5. }
  1. @Test
  2. public void sort() {
  3. List<Obj> list = Arrays.asList(
  4. new Obj("政府", null),
  5. new Obj("政府", new BigDecimal("1216.23")),
  6. new Obj("商业", new BigDecimal("123.23")),
  7. new Obj("PPD", new BigDecimal("123.23")),
  8. new Obj("合作", new BigDecimal("127.23")),
  9. new Obj(null, new BigDecimal("125.23")));
  10. List<String> sortList = Arrays.asList("PPD","政府","合作");
  11. List<Obj> result = list.stream().sorted(
  12. //先按照name排序(模拟需求的a属性排序)
  13. Comparator.comparing(Obj::getName,(x,y)-> {
  14. if(x == null && y != null){
  15. return 1;
  16. }else if(x !=null && y == null){
  17. return -1;
  18. }else if(x == null && y == null){
  19. return -1;
  20. } else {
  21. for(String sort : sortList){
  22. if(sort.equals(x) || sort.equals(y)){
  23. if(x.equals(y)){
  24. return 0;
  25. }else if(sort.equals(x)){
  26. return -1;
  27. }else{
  28. return 1;
  29. }
  30. }
  31. }
  32. return 0;
  33. }
  34. })).collect(Collectors.toList());
  35. System.out.println(result);
  36. }

1.实现

comparing方法有两种实现

方法1:只有一个参数,参数的类型是一个函数式接口 在这里插入图片描述

方法2:

问:这个方法中泛型是怎么传递的

1、list.stream()时,获取的stream流已经确定了泛型了,此时返回的对象为Stream<Obj>

2、Stream对象的sorted方法,需要比较器的类型需要是Obj.calss或者是Obj的父类

在这里插入图片描述

3、而我们这边调用了静态方法Comparator.comparing,静态方法中的泛型是根据传的参数中的类型来决定的

comparing

以上为个人经验,希望能给大家一个参考,也希望大家多多支持w3xue。

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号