将多个值关联到同一个 key的map(key可以重复的map)示例

在 Java 中,标准的 Map 接口要求 key 必须唯一 ,如果需要 key 可重复 且保持 插入顺序 的数据结构,可以使用以下方案:

1. 使用 List<Map.Entry<K, V>>

最直接的方式是用链表存储键值对,允许重复 key:

java 复制代码
import java.util.*;

// 创建可重复 key 的链表
List<Map.Entry<String, Integer>> list = new ArrayList<>();

// 添加元素(key 可重复)
list.add(new AbstractMap.SimpleEntry<>("key1", 1));
list.add(new AbstractMap.SimpleEntry<>("key1", 2)); // 重复 key
list.add(new AbstractMap.SimpleEntry<>("key2", 3));

// 遍历所有元素(保持插入顺序)
for (Map.Entry<String, Integer> entry : list) {
    System.out.println(entry.getKey() + " : " + entry.getValue());
}

// 查找特定 key 的所有值
list.stream()
    .filter(e -> "key1".equals(e.getKey()))
    .forEach(e -> System.out.println("key1 值: " + e.getValue()));

2. 使用 Guava 的 Multimap

Google Guava 库提供的 Multimap 可以将多个值关联到同一个 key:

java 复制代码
import com.google.common.collect.ArrayListMultimap;
import com.google.common.collect.Multimap;

// 创建可重复 key 的 Multimap(保持插入顺序)
Multimap<String, Integer> multimap = ArrayListMultimap.create();

// 添加元素(key 可重复)
multimap.put("key1", 1);
multimap.put("key1", 2); // 重复 key
multimap.put("key2", 3);

// 获取 key 对应的所有值(返回 Collection)
Collection<Integer> values = multimap.get("key1"); // 返回 [1, 2]

// 遍历所有键值对
multimap.entries().forEach(entry -> 
    System.out.println(entry.getKey() + " : " + entry.getValue())
);

3. 自定义 MultiMap

如果不想引入外部依赖,可以自定义一个包装类:

java 复制代码
import java.util.*;

public class MultiMap<K, V> {
    private final List<Map.Entry<K, V>> entries = new ArrayList<>();

    // 添加键值对(允许重复 key)
    public void put(K key, V value) {
        entries.add(new AbstractMap.SimpleEntry<>(key, value));
    }

    // 获取 key 对应的所有值
    public List<V> get(K key) {
        List<V> result = new ArrayList<>();
        for (Map.Entry<K, V> entry : entries) {
            if (Objects.equals(key, entry.getKey())) {
                result.add(entry.getValue());
            }
        }
        return result;
    }

    // 遍历所有键值对
    public void forEach(BiConsumer<K, V> action) {
        entries.forEach(entry -> action.accept(entry.getKey(), entry.getValue()));
    }

    // 其他方法(size、isEmpty 等)...
}

// 使用示例
MultiMap<String, Integer> multiMap = new MultiMap<>();
multiMap.put("key1", 1);
multiMap.put("key1", 2);
multiMap.forEach((k, v) -> System.out.println(k + " : " + v));

选择建议

  • 简单场景 :直接使用 List<Map.Entry<K, V>>
  • 需要丰富 API :引入 Guava 的 Multimap
  • 不想依赖外部库 :自定义 MultiMap

无论哪种方案,都能实现 key 可重复保持插入顺序 的需求。

相关推荐
东阳马生架构7 小时前
商品中心—6.商品考核系统的技术文档
java
晴空月明7 小时前
Java 内存模型与 Happens-Before 关系深度解析
java
皮皮林55111 小时前
SpringBoot 加载外部 Jar,实现功能按需扩展!
java·spring boot
rocksun11 小时前
认识Embabel:一个使用Java构建AI Agent的框架
java·人工智能
Java中文社群12 小时前
AI实战:一键生成数字人视频!
java·人工智能·后端
王中阳Go13 小时前
从超市收银到航空调度:贪心算法如何破解生活中的最优决策谜题?
java·后端·算法
shepherd11113 小时前
谈谈TransmittableThreadLocal实现原理和在日志收集记录系统上下文实战应用
java·后端·开源
维基框架13 小时前
Spring Boot 项目整合Spring Security 进行身份验证
java·架构
日月星辰Ace14 小时前
Java JVM 垃圾回收器(四):现代垃圾回收器 之 Shenandoah GC
java·jvm
天天摸鱼的java工程师15 小时前
商品详情页 QPS 达 10 万,如何设计缓存架构降低数据库压力?
java·后端·面试