Java stream sorted使用 Comparator 進(jìn)行多字段排序
摘要:介紹使用Java Stream流排序器Comparator對List集合進(jìn)行多字段排序的方法,包括復(fù)雜實(shí)體對象多字段升降序混合排序方法。
綜述
??Java 8 的 Stream 使用了函數(shù)式編程模式,人如其名,它可以被用來對集合或數(shù)組進(jìn)行鏈狀流式的排序、過濾和統(tǒng)計(jì)等操作,從而讓我們更方便的對集合或數(shù)組進(jìn)行操作。
??關(guān)于List排序,工作中,一般使用SQL中的order by進(jìn)行排序,但有時(shí)候使用Java代碼進(jìn)行排序,例如合并多個(gè)list對象的數(shù)據(jù)后,以年齡降序排列,這顯然是無法通過SQL語句搞定的,而一般的冒泡排序、希爾排序等需要手寫實(shí)現(xiàn),容易出錯(cuò),而且代碼量大,測試工作量自然不容小覷。這時(shí),就需要搬出Stream sort方法進(jìn)行排序,重寫其中的Comparator。
??本文重點(diǎn)介紹使用Java Stream流排序器Comparator對List集合進(jìn)行排序的技巧,包括復(fù)雜實(shí)體對象多字段升降序排序方法。
重寫類的Comparable接口
??重寫List中泛型Bean的compareTo方法實(shí)現(xiàn)排序,即流中泛型元素需實(shí)現(xiàn)Comparable接口,實(shí)現(xiàn)如下:
import lombok.Getter;
import lombok.Setter;
import lombok.ToString;
import java.io.Serializable;
/**
* 用戶實(shí)體
*/
@Getter
@Setter
@ToString
public class UserDTO implements Serializable, Comparable<UserDTO> {
private static final long serialVersionUID = -2618535482684811077L;
/**
* 用戶id
*/
private Long id;
/**
* 姓名
*/
private String name;
/**
* 年齡
*/
private Integer age;
/**
* 是否男士,true 是
*/
private Boolean isBoy;
/**
* 無參構(gòu)造器
*/
public UserDTO() {
System.out.println("無參構(gòu)造器");
}
private UserDTO(String userName) {
this.name = userName;
System.out.println("一個(gè)參數(shù)的構(gòu)造器,private");
}
public UserDTO(Long id, String userName, Integer age) {
this.id = id;
this.name = userName;
this.age = age;
}
public UserDTO(Long id, String userName, Integer age, Boolean isBoy) {
this.id = id;
this.name = userName;
this.age = age;
this.isBoy = isBoy;
}
@Override
public int compareTo(UserDTO o) {
return id.compareTo(o.getId());
}
}
??缺點(diǎn)是所有類都會(huì)使用這個(gè)排序規(guī)則,不適用于排序規(guī)則靈活多變的復(fù)雜業(yè)務(wù)場景。
使用Comparator排序
??使用stream的sorted(Comparator com)基于自定義規(guī)則排序,這需要為comparing 和thenComparing自定義Comparator排序器,以實(shí)現(xiàn)升序或者降序。接下來進(jìn)行案例分析的時(shí)候,默認(rèn)UserDTO沒有重寫類的Comparable接口。
sorted comparing 自然排序
??sorted 排序結(jié)果默認(rèn)升序排序,它根據(jù)comparing來實(shí)現(xiàn)。語法糖:
// 從類型T中提取Comparable排序?qū)傩裕⒎祷卦搶傩缘谋容^器Comparator<T>
static <T,U extends Comparable<? super U>> Comparator<T> comparing(Function<? super T,? extends U> keyExtractor)
// 從T類型對象提取U類型的排序字段,并返回一個(gè)根據(jù)此排序字段Comparator<T>
static <T,U> Comparator<T> comparing(Function<? super T,? extends U> keyExtractor, Comparator<? super U> keyComparator)
?? Function 是一個(gè)函數(shù)接口,包含一種 apply()方法,來實(shí)現(xiàn)方法調(diào)用。參數(shù)Function<? super T, ? extends U> keyExtractor表示輸入一個(gè) T 類型形參,輸出一個(gè) U 類型的對象。舉個(gè)例子,輸入一個(gè) UserDTO 對象返回其Integer類型屬性年齡(age)的數(shù)值:
Function<UserDTO, Integer> getUserAge = UserDTO::getAge;
使用默認(rèn)屬性排序:
list = list.stream().sorted().collect(Collectors.toList());
??下面是根據(jù)年齡升序排序的示例:
list = list.stream().sorted(Comparator.comparing(UserDTO::getAge))
.collect(Collectors.toList());
??如果想實(shí)現(xiàn)降序排列,可以使用Comparator 提供的reverseOrder() 方法
list = list.stream().sorted(Comparator.reverseOrder()).collect(Collectors.toList());
??下面是根據(jù)年齡降序排列的示例:
list = list.stream().sorted(Comparator.comparing(UserDTO::getAge).reversed())
.collect(Collectors.toList());
or
list = list.stream().sorted(Comparator.comparing(UserDTO::getAge, Comparator.reverseOrder()))
.collect(Collectors.toList());
??像Integer、Long等基本類型的包裝類已經(jīng)實(shí)現(xiàn)了Comparable接口,在使用sorted排序的時(shí)候,可以使用comparingInt、thenComparingInt、thenComparingLong等。
thenComparing 多字段排序
??在多于一個(gè)屬性排序的場景,可以結(jié)合 comparing 和 thenComparing進(jìn)行解決——先使用comparing進(jìn)行比較,再使用一個(gè)或者多個(gè)thenComparing進(jìn)行排序。語法糖:
//用另一個(gè)比較器 other 返回一個(gè)字典順序排序比較器
default Comparator<T> thenComparing(Comparator<? super T> other)
// 返回指定屬性的 Comparable順序比較器
default <U extends Comparable<? super U>> Comparator<T> thenComparing(Function<? super T,? extends U> keyExtractor)
// 返回一個(gè)根據(jù)T對象U類型字段字段排序的Comparator
default <U> Comparator<T> thenComparing(Function<? super T,? extends U> keyExtractor, Comparator<? super U> keyComparator)
??案例1:集合以泛型類的屬性一升序、屬性二升序排序:
Comparator<類> comparator = Comparator.comparing(類::屬性一).thenComparing(類::屬性二);
list=list.stream().sorted(comparator).collect(Collectors.toList());
??案例2:按用戶年齡升序,年齡相同時(shí)則按姓名升序:
List<UserDTO> sortedList=list.sorted(Comparator.comparing(UserDTO::getAge).thenComparing(UserDTO::getName))
.collect(Collectors.toList());
sortedList.stream().forEach(System.out::println);
??案例3:排序結(jié)果以屬性一降序,屬性二升序排列:
Comparator<類> comparator = Comparator.comparing(類::屬性一,Comparator.reverseOrder()).thenComparing(類::屬性二);
list=list.stream().sorted(comparator).collect(Collectors.toList());
??這里自定義了一個(gè)比較器對象,修改對象排序規(guī)則即可。如果某個(gè)屬性需要降序,則在comparing中聲明Comparator.reverseOrder(),例如:
Comparator<UserDTO> comparator = Comparator.comparing(UserDTO::getAge, Comparator.reverseOrder()).thenComparing(UserDTO::getName);
list=list.sorted(comparator).collect(Collectors.toList());
??當(dāng)然了,也可以把Comparator.reverseOrder()放到屬性二的位置,此時(shí)表示以屬性一升序、屬性二降序排列:
list=list.stream().sorted(Comparator.comparing(類::屬性一).thenComparing(類::屬性二,Comparator.reverseOrder()))
.collect(Collectors.toList());
注意事項(xiàng)
??1、降序排列時(shí),只需要在 comparator 末尾寫一個(gè) reversed(),不需要每個(gè)比較屬性都寫
Comparator<類> comparator1 =
Comparator.comparing(類::屬性一).thenComparing(類::屬性二).reversed();
??但是,不建議這樣寫,推薦如下語義更清晰的語法糖:
Comparator<類> comparator1 = Comparator.comparing(類::屬性一, Comparator.reverseOrder()).thenComparing(類::屬性二, Comparator.reverseOrder())
??2、構(gòu)建比較器時(shí)如果分多行,不能以如下形式定義,否則會(huì)排序不正確:
Comparator<類> comparator2 = Comparator.comparing(類::屬性一);
comparator2.thenComparing(類::屬性二);
但可以寫成
Comparator<類> comparator2 = Comparator.comparing(類::屬性一);
comparator2 = comparator2.thenComparing(類::屬性二);
??3、sorted()方法返回的結(jié)果集是一個(gè)新的對象,和被排序?qū)ο蟮囊貌灰粯印?/p>
??4、參與排序的類屬性需要注意是否為null。如果為null可以設(shè)置一個(gè)默認(rèn)值,或者使用如下方法解決:
//值為null的元素排在前面
Comparator<類> firstComparator = Comparator.nullsFirst(Comparator.comparing(類::屬性一));
//所有的空元素將被排在最后,不影響非空元素排序
Comparator<類> lastComparator = Comparator.nullsLast(Comparator.comparing(類::屬性一));
Reference
Buy me a coffee. ?Get red packets.
浙公網(wǎng)安備 33010602011771號(hào)