關於java8 的stream排序用法這裏不做多說,這裏介紹下曾經在多字段排序時遇到過的一個坑。
需求:需要根據id去分組,然後取出每組中行號最大的一個對象值。
想到可以利用stream的多字段排序,先按id去排,再看行號去排,demo代碼如下:
class Tt{
private int id;
private int line;
public Tt(int id, int line) {
this.id = id;
this.line = line;
}
public int getId() {
return id;
}
public void setId(int id) {
this.id = id;
}
public int getLine() {
return line;
}
public void setLine(int line) {
this.line = line;
}
@Override
public String toString() {
return "Tt{" +
"id=" + id +
", line=" + line +
'}';
}
}
對象代碼如上,只提供id和行號字段,做演示用:
測試代碼如下:
public static void main(String[] args) {
List<Tt> list = new ArrayList<>();
list.add(new Tt(1,2));
list.add(new Tt(2,2));
list.add(new Tt(5,2));
list.add(new Tt(5,1));
list.add(new Tt(4,2));
list.add(new Tt(7,2));
list.add(new Tt(3,2));
List<Tt> sortedList = list.stream() .sorted(Comparator.comparing(Tt::getId)).sorted(Comparator.comparing(Tt::getLine))
.collect(Collectors.toList());
System.err.println(sortedList);
}
根據理論,應該是先根據id去排好序,再根據行號去排號序,效果應該是,按id順序打印的。結果如下:
發現其實並不是想像的效果,結果是按後者順序打印的。
由此推斷出,當需要用到stram多條件排序的時候,需要最後排序的字段需要放在前面排,改後代碼如下:
List<Tt> sortedList = list.stream() .sorted(Comparator.comparing(Tt::getLine)).sorted(Comparator.comparing(Tt::getId))
.collect(Collectors.toList());
效果如下:
達到預期,問題解決!