java—如何按两个字段对数组列表排序

m0rkklqb  于 2021-07-09  发布在  Java
关注(0)|答案(3)|浏览(530)

我有一个双数组的列表,我想按第一个和最后一个字段排序。到目前为止,我只能按数组的1个元素对它进行排序。
真实状态:

0         1         2
-------  -------  -------
78       100        0
78       100        1
0        100        0
104      100        1

预期:

0         1         2
-------  -------  -------
0        100       0
78       100       1
78       100       0
101      100       1

我想根据数组第一个元素的值对字段进行排序。如果第一和第二的值相等,我想根据第三个元素排序,其中第一个元素应该是1,然后是0(只有1和0的值)

List<Double[]> splitList = new ArrayList<>();
Double[] tmp1 = { 78d, 100d, 0d };
Double[] tmp2 = { 78d, 100d, 1d };
Double[] tmp3 = { 0d, 100d, 0d };
Double[] tmp4 = { 104d, 100d, 1d };
splitList.add(tmp1);
splitList.add(tmp2);
splitList.add(tmp3);
splitList.add(tmp4);
splitList.sort(Comparator.comparingDouble(a -> a[0]));

这个根据第一个元素来分类。我找到了按两个元素排序的解决方案https://stackoverflow.com/a/26865122/9774735 所以我试了一下:

splitList.sort(Comparator.comparingDouble(a -> a[0]).thenComparingDouble(b -> b[2]));

它给了我一个错误:

Multiple markers at this line
    - The type of the expression must be an array type but it resolved 
     to Object
    - The type of the expression must be an array type but it resolved

如何比较数组列表?

ljo96ir5

ljo96ir51#

试试这个:

splitList.sort((a,b) -> {
        int r = a[0].compareTo(b[0]);
        if (r == 0) {
            r = a[2].compareTo(b[2]);
        }
        return r;
    });
5cnsuln7

5cnsuln72#

如果您使用java 8或下一版本,您可以使用:

splitList.stream()
            .sorted(Comparator.comparing((Double[] array) -> array[0])
                    .thenComparing(Comparator.comparing((Double[] array2) -> array2[2]).reversed()));

要在控制台中打印:

splitList.stream()
            .sorted(Comparator.comparing((Double[] array) -> array[0])
                    .thenComparing(Comparator.comparing((Double[] array2) -> array2[2]).reversed()))
            .forEach(array -> System.out.println(Arrays.toString(array)));

或者您可以在列表中收集,然后也可以打印列表:

List<Double[]> list = splitList.stream()
            .sorted(Comparator.comparing((Double[] array) -> array[0])
                    .thenComparing(Comparator.comparing((Double[] array2) -> array2[2]).reversed()))
            .collect(Collectors.toList());

for (Double[] array: list) {
    System.out.println(Arrays.toString(array));
}

祝你好运!

m528fe3b

m528fe3b3#

似乎java编译器在推断泛型类型方面有问题,您可以使用一些选项来解决此问题:
在模式中使用类型提示 Class.<GenericType>.method(Arguments) :

splitList.sort(Comparator.<Double[]>comparingDouble(a -> a[0]).thenComparingDouble(b -> b[2]));

声明lambda参数类型(第一次声明就足够了):

splitList.sort(Comparator.comparingDouble((Double[] a) -> a[0]).thenComparingDouble(b -> b[2]));

在阅读了您的评论之后,您想反转上一个比较,可以这样做:

Comparator.<Double[]>comparingDouble(a -> a[0])
   .thenComparing(Comparator.<Double[]>comparingDouble(b -> b[2]).reversed())

这是相当混乱的,你最好用这样的东西:

splitList.sort((a, b) -> {
    int c = Double.compare(a[0], b[0]);
    return c == 0 ? Double.compare(b[2], a[2]) : c;
});

相关问题