Java8 Collectors.toMap的兩個大坑


  1. Collectors.toMap()方法的正常使用示例
List<StudentDTO> studentDTOS = Lists.newArrayList();
studentDTOS.add(new StudentDTO(1,"xixi"));
studentDTOS.add(new StudentDTO(2,"houhou"));
studentDTOS.add(new StudentDTO(3,"maomi"));
Map<Integer, String> collect = studentDTOS.stream().collect(
	Collectors.toMap(StudentDTO::getStudentId, StudentDTO::getStudentName));
System.out.println(JSON.toJSON(collect)); // {"1":"xixi","2":"houhou","3":"maomi"}

一. 坑1:Duplicate Key時拋出IllegalStateException異常

1. 概述

  • 按照常規Java的Map思維,往一個map裏put一個已經存在的key,會把原有的key對應的value值覆蓋。
  • 但Java8中的Collectors.toMap()卻不是這樣。當key重複時,該方法默認會拋出IllegalStateException異常。

2. 大坑復現

public void streamToMap1() {
    List<StudentDTO> studentDTOS = Lists.newArrayList();
    studentDTOS.add(new StudentDTO(1,"xixi"));
    studentDTOS.add(new StudentDTO(1,"houhou"));
    studentDTOS.add(new StudentDTO(3,"maomi"));
    Map<Integer, String> collect = studentDTOS.stream()
    	.collect(Collectors.toMap(StudentDTO::getStudentId, StudentDTO::getStudentName));
    System.out.println(JSON.toJSON(collect)); 
}
  • 輸出結果
    在這裏插入圖片描述

3. 大坑解決

  1. 法1:將toMap方法修改成如下形式,這樣就可以使用新的value覆蓋原有value。
studentDTOS.stream().collect(Collectors.toMap(StudentDTO::getStudentId, 
					StudentDTO::getStudentName,(oldValue, newValue) -> newValue));
  • 輸出結果:{"1":"houhou","3":"maomi"}
  1. 法2:如果需要保留同一個key下所有的值,則可以對value做簡單的拼接,如下:
studentDTOS.stream().collect(Collectors.toMap(StudentDTO::getStudentId, 
					StudentDTO::getStudentName,(oldValue, newValue) -> oldValue + "," + newValue));
  • 輸出結果:{"1":"xixi,houhou","3":"maomi"}

二. 坑2:value爲空時拋出NullPointerException異常

1. 概述

  • 當要轉化的map的value值中包含空指針時, 會拋出NullPointerException異常。

2. 大坑復現

public void streamToMap2() {
    List<StudentDTO> studentDTOS = Lists.newArrayList();
    studentDTOS.add(new StudentDTO(1,"xixi"));
    studentDTOS.add(new StudentDTO(2,"houhou"));
    studentDTOS.add(new StudentDTO(3,null));
    Map<Integer, String> collect = studentDTOS.stream().collect(Collectors
    .toMap(StudentDTO::getStudentId, StudentDTO::getStudentName));
    System.out.println(JSON.toJSON(collect));
}
  • 輸出結果
    在這裏插入圖片描述

3. 大坑解決

3.1 法1:value值判空設置

  1. 說明:如果是null,則設置成一個特定值。
studentDTOS.stream().collect(Collectors.toMap(StudentDTO::getStudentId, studentDTO 
					-> studentDTO.getStudentName()==null?"":studentDTO.getStudentName()));

輸出結果:{"1":"xixi","2":"houhou","3":""}

3.2 法2:使用collect(Supplier<R> supplier, BiConsumer<R, ? super T> accumulator, BiConsumer<R, R> combiner)方法構建

  1. 說明:該方法允許null值。
Map<Integer, String> collect = studentDTOS.stream().collect(HashMap::new, 
		(n, v) -> n.put(v.getStudentId(), v.getStudentName()), HashMap::putAll);
for(Map.Entry<Integer, String> entry:collect.entrySet()){
    System.out.println(entry.getKey()+"="+entry.getValue());
}
  • 輸出結果
1=xixi
2=houhou
3=null

3.3 使用Optional對值進行包裝

Map<Integer, Optional<String>> collect = studentDTOS.stream().collect(Collectors
	.toMap(StudentDTO::getStudentId,
	studentDTO -> Optional.ofNullable(studentDTO.getStudentName())));
	
for(Map.Entry<Integer, Optional<String>> entry:collect.entrySet()){
    System.out.println(entry.getKey()+"="+entry.getValue().orElse(""));
}
  • 輸出結果
1=xixi
2=houhou
3=

參考資料

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章