Java Multimap实现类与操作的具体示例

2025-06-09 16:50

本文主要是介绍Java Multimap实现类与操作的具体示例,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

《JavaMultimap实现类与操作的具体示例》Multimap出现在Google的Guava库中,它为Java提供了更加灵活的集合操作,:本文主要介绍JavaMultimap实现类与操作的...

一、Multimap 概述

Multimap 是 Google Guava 库中提供的一种集合类型,它扩展了传统的 Map 概念,允许一个键对应多个值。与标准的 Map<K, List<V>> 或 Map<K, Set<V>> 相比,Multimap 提供了更简洁的 API 和更强大的功能。

Multimap 主要特点:

  • 一个键可以映射到多个值
  • 避免了手动管理值集合的麻烦
  • 提供了丰富的视图集合
  • 支持不可变实现

二、Multimap 实现类

Guava 提供了多种 Multimap 实现,每种实现都有不同的特性和使用场景。

1. ListMultimap 系列

特点:值以 List 形式存储,允许重复值,保留插入顺序

实现类描述是否线程安全
ArrayListMultimap使用 ArrayList 作为值集合
LinkedListMultimap使用 LinkedList 作为值集合
ImmutableListMultimap不可变实现

2. SetMultimap 系列

特点:值以 Set 形式存储,不允许重复值

实现类描述是否线程安全
HashMultimap使用 HashSet 作为值集合
LinkedHashMultimap使用 LinkedHashSet 作为值集合,保留插入顺序
TreeMultimap使用 TreeSet 作为值集合,按键和值排序
ImmutableSetMultimap不可变实现

3. 其他实现

实现类描述是否线程安全
Multimaps.synchronizedMultimap同步包装器,使任何 Multimap 线程安全
Multimaps.unmodifiableMultimap不可修改视图

三、Multimap 基本操作示例

1. 创建 Multimap

// 创建ArrayListMultimap
ListMultimap<String, String> listMultimap = ArrayListMultimap.create();

// 创建HashMultimap
SetMultimap<String, Integer> setMultimap = HashMultimap.create();

// 创建不可变Multimap
ImmutableListMultimap<String, String> immutableMultimap = ImmutableListMultimap.of(
    "key1", "value1",
    "key1", "value2",
    "key2", "value3"
);

2. 添加元素

ListMultimap<String, String> multimap = ArrayListMultimap.create();

// 添加单个元素
multimap.put("fruit", "apple");
multimap.put("fruit", "banana");
multimap.put("fruit", "orange");
multimap.put("vegetable", "carrot");

// 添加多个元素
multimap.putAll("fruit", Arrays.asList("pear", "grape"));
multimap.putAll("vegetable", Arrays.asList("potato", "tomato"));

3. 获取元素

// 获取某个键的所有值
List<String> fruits = multimap.get("fruit"); // [apple, banana, orange, pear, grape]

// 获取第一个值
String firstFruit = multimap.get("fruit").get(0); // apple

// 检查键是否存在
boolean hasFruit = multimap.containsKey("fruit"); // true

// 检查键值对是否存在
boolean hasApple = multimap.containsEntry("fruit", "apple"); // true

4. 删除元素

// 删除键的所有值
multimap.removeAll("fruit"); // 返回被删除的值列表

// 删除特定键值对
multimap.remove("vegetable", "tomato"); // 返回boolean表示是否删除成功

// 清空所有元素
multimap.clear();

5. 视图操作

// 获取所有键的集合(去重)
Set<String> keys = multimap.keySet();

// 获取所有值的集合(不去重)
Collection<String> values = multimap.values();

// 获取键值对集合
Collection<Map.Entry<String, String>> entries = multimap.entries();

// 将Multimap转换为Map<K, Collection<V>>
Map<String, Collection<String>> mapView = multimap.asMap();

四、不同实现类的具体示例

1. ArrayListMultimap 示例

// 创建ArrayListMultimap
ListMultimap<String, Integer> scores = ArrayListMultimap.create();

// 添加元素
scores.put("Alice", 90);
scores.put("Alice", 85);
scores.put("Bob", 75);
scores.put("Bob", 80);
scores.put("Bob", 82);

// 获取元素
List<Integer> aliceScores = scores.get("Alice"); // [90, 85]
List<Integer> bobScores = scores.get("Bob"); // [75, 80, 82]

// 允许重复值
scores.put("Alice", 90);
List<Integer> newAliceScores = scores.get("Alice"); // [90, 85, 90]

2. HashMultimap 示例

// 创建HashMultimap
SetMultimap<String, String> tags = HashMultimap.create();

// 添加元素
tags.put("article1", "tech");
tags.put("article1", "Java");
tags.put("article1", "programming");
tags.put("article2", "design");
tags.put("article2", "ui");

// 尝试添加重复值
tags.put("article1", "java"); // 不会有任何效果

// 获取元素
Set<String> artidgNiROcle1Tags = tags.get("article1"); // [tech, java, programming]
Set<String> article2pythonTags = tags.get("article2"); // [design, ui]

3. TreeMultimap 示例

// 创建TreeMultimap(按键和值排序)
TreeMultimap<String, Integer> sortedScores = TreeMultimap.create();

// 添加元素(乱序)
sortedScores.put("Bob", 80);
sortedScores.put("Alice", 90);
sortedScores.put("Bob", 75);
sortedScores.put("Alice", 85);

// 获取元素(自动排序)
SortedSet<Integer> aliceSortedScores = sortedScores.get("Alice"); // [85, 90]
SortedSet<Integer> bobSortedScores = sortedScores.get("Bob"); // [75, 80]

// 整个Multimap也是按键排序的
System.out.println(sortedScores); // {Alice=[85, 90], Bob=[75, 80]}

4. ImmutableListMultimap 示例

// 创建不可变Multimap
ImmutableListMultimap<String, String> immutableMap = ImmutableListMultimapChina编程.<String, String>builder()
    .put("colors", "red")
    .put("colors", "green")
    .put("colors", "blue")
    .put("shapes", "circle")
    .put("shapes", "square")
    .build();

// 尝试修改会抛出UnsupportedOperationException
// immutableMap.put("colors", "yellow"); // 错误!

// 安全地获取数据
List<String> colors = immutableMap.get("colors"); // [red, green, blue]

五、高级操作与工具方法

1. 使用 Multimaps 工具类

// 从Map<K, Collection<V>>创建Multimap
Map<String, List<Integer>> map = new HashMap<>();
map.put("a", Arrays.asList(1, 2, 3));
map.put("b", Arrays.asList(4, 5));
ListMultimap<String, Integer> multimap = Multimaps.forMap(map);

// 转换Multimap的值类型
ListMultimap<String, String> stringMultimap = ArrayListMultimap.create();
stringMultimap.put("key", "1");
stringjavascriptMultimap.put("key", "2");
ListMultimap<String, Integer> intMultimap = Multimaps.transformValues(stringMultimap, Integer::valueOf);

// 反转Multimap(键值互换)
SetMultimap<Integer, String> inverted = Multimaps.invertFrom(stringMultimap, HashMultimap.create());

2. 过滤操作

ListMultimap<String, Integer> scores = ArrayListMultimap.create();
scores.put("Alice", 90);
scores.put("Alice", 85);
scores.put("Bob", 75);
scores.put("Bob", 80);

// 过滤出分数大于80的条目
Multimap<String, Integer> highScores = Multimaps.filterEntries(scores, 
    entry -> entry.getValue() > 80);

System.out.println(highScores); // {Alice=[90, 85]}

3. 同步包装

ListMultimap<String, String> unsafeMultimap = ArrayListMultimap.create();

// 创建线程安全版本
ListMultimap<String, String> safeMultimap = Multimaps.synchronizedListMultimap(unsafeMultimap);

// 现在可以安全地在多线程环境中使用
safeMultimap.put("key", "value");

六、性能比较与选择指南

实现类键存储值存储允许重复值顺序保证典型用途dgNiRO
ArrayListMultimapHashMapArrayList插入顺序需要保留插入顺序且允许重复值的场景
LinkedListMultimapLinkedHashMapLinkedList插入顺序需要频繁在中间插入/删除的场景
HashMultimapHashMapHashSet需要快速查找且不需要重复值的场景
LinkedHashMultimapLinkedHashMapLinkedHashSet插入顺序需要保留插入顺序且不允许重复值的场景
TreeMultimapTreeMapTreeSet排序顺序需要按键和值排序的场景
ImmutableListMultimap不可变不可变构造顺序需要不可变集合的场景
ImmutableSetMultimap不可变不可变构造顺序需要不可变集合且不允许重复值的场景

选择建议

  • 如果需要允许重复值 - 选择 ListMultimap 实现
  • 如果需要快速查找且不允许重复值 - 选择 SetMultimap 实现
  • 如果需要排序功能 - 选择 TreeMultimap
  • 如果需要线程安全 - 使用不可变实现或同步包装器
  • 如果数据不常变化 - 优先考虑不可变实现

七、常见问题解答

Q1: Multimap 和 Map<K, List> 有什么区别?

A1: 主要区别在于:

  • Multimap 提供了更简洁的 API,不需要手动管理值集合
  • Multimap 提供了丰富的视图方法(如 entries(), keys(), values())
  • Multimap 隐藏了实现细节,可以更灵活地切换底层实现
  • Multimap 的方法更直观,如 put() 直接添加元素而不需要先检查是否存在集合

Q2: 如何将 Multimap 转换为传统的 Map?

A2: 可以使用 asMap() 方法:

ListMultimap<String, String> multimap = ArrayListMultimap.create();
multimap.put("key", "value1");
multimap.put("key", "value2");

Map<String, Collection<String>> map = multimap.asMap();

Q3: Multimap 是线程安全的吗?

A3: 大多数 Multimap 实现不是线程安全的,除了:

  • 不可变实现(ImmutableListMultimap, ImmutableSetMultimap)
  • 使用 Multimaps.synchronizedMultimap() 包装的 Multimap

Q4: 如何统计 Multimap 中每个键对应的值数量?

A4: 可以使用 Multimap 的 keys() 方法结合 Multisets:

ListMultimap<String, String> multimap = ArrayListMultimap.create();
// 添加元素...

Multiset<String> counts = HashMultiset.create(multimap.keys());
System.out.println(counts); // 显示每个键的出现次数

八、总结

Guava 的 Multimap 提供了一种优雅的方式来处理键到多个值的映射关系,比传统的 Map<K, Collection<V>> 更加方便和强大。通过选择合适的实现类,可以满足各种不同的业务需求,包括是否需要允许重复值、是否需要保持顺序、是否需要排序等功能。

在实际开发中,Multimap 特别适用于以下场景:

  • 标签系统(一个项目有多个标签)
  • 学生成绩记录(一个学生有多门成绩)
  • 反向索引(一个单词出现在多个文档中)
  • 分组统计(按类别分组记录)

掌握 Multimap 的使用可以显著简化代码,提高开发效率,是 Java 开发者工具箱中不可或缺的工具之一。

到此这篇关于Java Multimap实现类与操作具体示例的文章就介绍到这了,更多相关Java Multimap类操作示例内容请搜索编程China编程(www.chinasem.cn)以前的文章或继续浏览下面的相关文章希望大家以后多多支持China编程(www.chinasem.cn)!

这篇关于Java Multimap实现类与操作的具体示例的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1154973

相关文章

SpringBoot连接Redis集群教程

《SpringBoot连接Redis集群教程》:本文主要介绍SpringBoot连接Redis集群教程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1. 依赖2. 修改配置文件3. 创建RedisClusterConfig4. 测试总结1. 依赖 <de

SpringBoot+Redis防止接口重复提交问题

《SpringBoot+Redis防止接口重复提交问题》:本文主要介绍SpringBoot+Redis防止接口重复提交问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不... 目录前言实现思路代码示例测试总结前言在项目的使用使用过程中,经常会出现某些操作在短时间内频繁提交。例

CSS3打造的现代交互式登录界面详细实现过程

《CSS3打造的现代交互式登录界面详细实现过程》本文介绍CSS3和jQuery在登录界面设计中的应用,涵盖动画、选择器、自定义字体及盒模型技术,提升界面美观与交互性,同时优化性能和可访问性,感兴趣的朋... 目录1. css3用户登录界面设计概述1.1 用户界面设计的重要性1.2 CSS3的新特性与优势1.

SpringSecurity显示用户账号已被锁定的原因及解决方案

《SpringSecurity显示用户账号已被锁定的原因及解决方案》SpringSecurity中用户账号被锁定问题源于UserDetails接口方法返回值错误,解决方案是修正isAccountNon... 目录SpringSecurity显示用户账号已被锁定的解决方案1.问题出现前的工作2.问题出现原因各

Qt 设置软件版本信息的实现

《Qt设置软件版本信息的实现》本文介绍了Qt项目中设置版本信息的三种常用方法,包括.pro文件和version.rc配置、CMakeLists.txt与version.h.in结合,具有一定的参考... 目录在运行程序期间设置版本信息可以参考VS在 QT 中设置软件版本信息的几种方法方法一:通过 .pro

Java继承映射的三种使用方法示例

《Java继承映射的三种使用方法示例》继承在Java中扮演着重要的角色,它允许我们创建一个类(子类),该类继承另一个类(父类)的所有属性和方法,:本文主要介绍Java继承映射的三种使用方法示例,需... 目录前言一、单表继承(Single Table Inheritance)1-1、原理1-2、使用方法1-

Oracle 数据库数据操作如何精通 INSERT, UPDATE, DELETE

《Oracle数据库数据操作如何精通INSERT,UPDATE,DELETE》在Oracle数据库中,对表内数据进行增加、修改和删除操作是通过数据操作语言来完成的,下面给大家介绍Oracle数... 目录思维导图一、插入数据 (INSERT)1.1 插入单行数据,指定所有列的值语法:1.2 插入单行数据,指

HTML5实现的移动端购物车自动结算功能示例代码

《HTML5实现的移动端购物车自动结算功能示例代码》本文介绍HTML5实现移动端购物车自动结算,通过WebStorage、事件监听、DOM操作等技术,确保实时更新与数据同步,优化性能及无障碍性,提升用... 目录1. 移动端购物车自动结算概述2. 数据存储与状态保存机制2.1 浏览器端的数据存储方式2.1.

基于 HTML5 Canvas 实现图片旋转与下载功能(完整代码展示)

《基于HTML5Canvas实现图片旋转与下载功能(完整代码展示)》本文将深入剖析一段基于HTML5Canvas的代码,该代码实现了图片的旋转(90度和180度)以及旋转后图片的下载... 目录一、引言二、html 结构分析三、css 样式分析四、JavaScript 功能实现一、引言在 Web 开发中,

Spring @Scheduled注解及工作原理

《Spring@Scheduled注解及工作原理》Spring的@Scheduled注解用于标记定时任务,无需额外库,需配置@EnableScheduling,设置fixedRate、fixedDe... 目录1.@Scheduled注解定义2.配置 @Scheduled2.1 开启定时任务支持2.2 创建