我们在利用java8 Lambda 表达式将集合中对象的属性转成Map时就会出现 Duplicate key xxxx , 说白了也就是key 重复了!案例如下:
@Getter
@Setter
@AllArgsConstructor
public class Student{
private String className;
private String studentName;
public static void main(String[] args) {
List<Student> list = new ArrayList<>();
list.add(new Student("一年级二班", "小明"));
list.add(new Student("一年级二班", "小芳"));
list.add(new Student("一年级二班", "小华"));
list.add(new Student("一年级三班", "翠花"));
list.add(new Student("一年级三班", "香兰"));
// 集合中对象属性转map
Map<String, String> map = list.stream().collect(Collectors.toMap(Student :: getClassName, Student :: getStudentName));
System.out.println(map);
}
}
此时将对象的 班级名称为 key 学生名称为 value,但运行时出现了多个相同的key ,此时编译器就会抛出
Duplicate key xxxx
解决方案如下:
我们需要使用toMap的另外一个重载的方法!
Collectors.toMap(keyMapper, valueMapper, mergeFunction)
前两两个参数都是与之前一样 key 和 value得取值属性, 第三个参数是当key 发生重复时处理的方法,注释上的解释如下:
一种合并函数,用于解决两者之间的冲突与提供的相同键相关联的值到{@link Map#merge(Object, Object, BiFunction)}
该合并函数有两个参数,第一个参数为当前重复key 之前对应的值,第二个为当前重复key 现在数据的值。
1、重复时采用后面的value 覆盖前面的value
Map<String, String> map = list.stream().collect(Collectors.toMap(Student :: getClassName, Student :: getStudentName,
(value1, value2 )->{
return value2;
}));
输出:
{一年级三班=香兰, 一年级二班=小华}
也可以简写成这样:
Map<String, String> map = list.stream().collect(Collectors.toMap(Student :: getClassName, Student :: getStudentName,
(key1 , key2)-> key2 ));
2、重复时将之前的value 和现在的value拼接或相加起来;
Map<String, String> map = list.stream().collect(Collectors.toMap(Student :: getClassName, Student :: getStudentName,
(key1 , key2)-> key1 + "," + key2 ));
输出:
{一年级三班=翠花,香兰, 一年级二班=小明,小芳,小华}
3、将重复key的数据变成一个集合!
Map<String, List<String>> map = list.stream().collect(Collectors.toMap(Student :: getClassName,
// 此时的value 为集合,方便重复时操作
s -> {
List<String> studentNameList = new ArrayList<>();
studentNameList.add(s.getStudentName());
return studentNameList;
},
// 重复时将现在的值全部加入到之前的值内
(List<String> value1, List<String> value2) -> {
value1.addAll(value2);
return value1;
}
));
输出:
{一年级三班=[翠花, 香兰], 一年级二班=[小明, 小芳, 小华]}
总结:
这几个办法都是基于toMap重载方法第三个参数来实现的!至于哪个方法最好,我觉得应该取决于具体业务!