list用stream流转map报key重复

我们在利用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的数据变成一个集合!

java 复制代码
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重载方法第三个参数来实现的!至于哪个方法最好,我觉得应该取决于具体业务!

相关推荐
稚辉君.MCA_P8_Java13 分钟前
豆包 Java的23种设计模式
java·linux·jvm·设计模式·kubernetes
tanyongxi6614 分钟前
C++ 特殊类设计与单例模式解析
java·开发语言·数据结构·c++·算法·单例模式
遗憾皆是温柔16 分钟前
24. 什么是不可变对象,好处是什么
java·开发语言·面试·学习方法
midsummer_woo21 分钟前
基于springboot的IT技术交流和分享平台的设计与实现(源码+论文)
java·spring boot·后端
Peter(阿斯拉)32 分钟前
[Java性能优化]_[时间优化]_[字符串拼接的多种方法性能分析]
java·性能优化·stringbuilder·string·字符串拼接·stringbuffer·时间优化
水痕012 小时前
gin结合minio来做文件存储
java·eureka·gin
寒士obj2 小时前
Spring事物
java·spring
柯南二号3 小时前
【Java后端】Spring Boot 集成 MyBatis-Plus 全攻略
java·spring boot·mybatis
桦说编程10 小时前
Java 中如何创建不可变类型
java·后端·函数式编程
lifallen10 小时前
Java Stream sort算子实现:SortedOps
java·开发语言