java8 对象转 Map 时重复 key Duplicate key xxxx

我们在利用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 ));
  1. 重复时将之前的value 和现在的value拼接或相加起来;
Map<String, String> map = list.stream().collect(Collectors.toMap(Student :: getClassName, Student :: getStudentName, 
	(key1 , key2)-> key1 + "," + key2 ));
 
输出:
{一年级三班=翠花,香兰, 一年级二班=小明,小芳,小华}
  1. 将重复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重载方法第三个参数来实现的!至于哪个方法最好,我觉得应该取决于具体业务!
### Java 8 中将对象换为 Map 的最佳实践 在Java 8中,利用流(Stream)API可以高效地完成这一任务。对于给定的对象列表,如果目标是基于某个属性作为键而另一个属性作为值构建映射,则可以通过`Collectors.toMap()`方法实现这一点[^1]。 下面是一个具体的例子,假设有如下类定义: ```java public class User { private String id; private String name; public User(String id, String name) { this.id = id; this.name = name; } // Getters and setters... } ``` 现在拥有一个User类型的列表,并希望依据id字段创建对应的映射关系,即`Map<String, User>`形式存储数据,那么可采用以下方式: ```java import java.util.*; import java.util.stream.Collectors; // 假设users是我们已有的用户列表 List<User> users = Arrays.asList( new User("u001", "Alice"), new User("u002", "Bob") ); // 换逻辑 Map<String, User> userById = users.stream() .collect(Collectors.toMap(User::getId, u -> u)); ``` 需要注意的是,在实际应用过程中可能会遇到重复键的情况,这会抛出异常如`IllegalStateException: Duplicate key ...`。为了避免这种情况发生,可以在调用`toMap`提供第三个参数——用于解决冲突的合并函数[^4]。 例如,当存在相同ID的不同记录可以选择保留任意一条或执行特定业务逻辑决定最终保存哪条记录: ```java .collect(Collectors.toMap( User::getId, Function.identity(), (existing, replacement) -> existing // 或者自定义合并策略 )); ``` 此外,若需进一步处理复杂结构的数据集或将其他格式(比如JSON字符串)解析成Map类型,还可以借助第三方库如Jackson来辅助完成这些工作[^2]。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值