老司机阿粉带你玩转 Guava 集合类(二)

简介: 日常开发中,阿粉经常需要用到 Java 提供集合类完成各种需求。Java 集合类虽然非常强大实用,但是提供功能还是有点薄弱。 举个例子,阿粉最近接到一个需求,从输入一个文档中,统计一个关键词出现的次数。代码如下

Multimap

阿粉有时会在业务需求中使用 Map<String,List<Integer>实现下面的需求。

1a->[1,2,3] b->4,c->[6,5]

使用 Map + List 这种结构比较笨拙,并且代码实现也比较繁琐。Multimap 正式 Guava 中解决这种问题的新出的一个雷。

使用 Multimap 实现代码如下:

67.jpg

这里阿粉使用 Multimap 子类 HashMultimap,其行为类似为 Map<K,Set<V>>,也就是说 Value 对应的集合内部元素不能重复。如果需要保存的重复的元素我们可以使用 ArrayListMultimap

Multimap还有其他子类,如图所示:

68.jpg

来源于 Github

BiMap

BiMap 可以用来实现键值对的双向映射需求,这样我们就可以通过 Key 查找对对应的 Value,也可以使用 Value 查找对应的 Key

这个需求如果使用 Map 实现,我们就不得不使用两个 Map,维护双向关系,并且任何改动还要保持同步。

69.jpg

使用 BiMap 修改上面的代码:

70.jpg

这里需要注意,BiMap#put方法不能加入重复元素, 若加入,将会抛错。如果若特定值一定要替换,可以使用 BiMap#forcePut代替。

敲黑板,这个知识点记下来。阿粉使用过程中,就踩过这个坑。

同样的 BiMap 也有各种实现类:

71.jpg

来源于 Github

其他扩展集合类

Guava 另外还提供其他集合类,不过这些类使用起来有点复杂,阿粉还未在业务代码中使用过,这里简单提下,感兴趣同学可以深入了解一下。

  • Table
  • ClassToInstanceMap
  • RangeSet
  • RangeMap

集合工具类

除了上面提到的新集合类以外,Guava 提供通用的工具类:

72.jpg

来源于 Github

这些工具类需对使用的方法,我们可以快速创建集合,分割集合,转化集合等。

快速创建集合实例

使用工具类,我们可以快速创建集合。例如:

1List<String> list=Lists.newArrayList();
2Set<String> set=Sets.newHashSet();
3Map<String,String> map=Maps.newHashMap();

相比于 new 集合方法,Guava 方法创建方式更加简单。

1List<String> list=new ArrayList<String>();
2Set<String> set=new HashSet<String>();
3Map<String,String> map=new HashMap<String, String>();

Guava 工具类智能推导 List 泛型,再也不用两侧都重复写泛型了。

另外还可以指定集合类的初始化大小。

73.jpg

Lists.transform

Lists#transform方法可以替代繁琐 for 循环,将元素转化,创建一个新集合类。

74.jpg

不过使用这个方法我们要注意一点。

Lists#transform 内部使用懒加载的机制,只有在调用获取的元素的时候,如 result.get 才会真正使用 Function 从源 List 获取元素,做相应的转化。每次获取元素都将会使用 function 进行转化。

所以使用 Lists#transform 得到 List 仅仅只是源 List 一个视图,任何对源 List 的元素修改,都将会被反应到创建之后的 List 。任何对创建之后 List 中的元素进行修改,都不会生效。下次再次读取元素时,将会发现相应修改的丢失了。。。

阿粉之前就踩过这个坑,如果你有这种需求,可以使用以下方式创建一个新集合:

75.jpg

JDK8 之前版本,阿粉经常使用该方法转化 List 中的元素。不过你如果使用 JDK8,阿粉还是推荐使用 Stream 流式编程。

交集并集差集

Sets 提供几个方法,可以快速求出两个 Set 集合的交集,并集以及差集。

不可变集合

不可变(Immutable)集合,顾名思义集合不可以被修改。初始创建不可变集合时吗,需要传入数据源,创建完成之后,集合就再也不能修改,增加,删除元素,否则将会报错。

这是一种防御性策略,防止集合在后续操作中被修改,从而引发问题。

不可变集合优点在于:

  • 由于不可变集合仅仅只能读,多线程并发天然安全
  • 由于不可变集合固定不变,可以将其当做常量安全,不用单线其他人修改
  • 不可变集合占用更少内存空间
  • 不可变集合不可以被修改,所以不用担心其他程序任意修改集合

Guava 不可变集合支持 JDK 所有集合接口:

76.jpg

我们可以使用如下几种方式创建不可变集合,以 ImmutableList 为例:

77.jpg

总结

这篇文章阿粉带大家学习开源工具 Guava 集合的相关类使用方法,日常开发中我们善于使用这些工具类,不要自己重复造轮子。

本篇文章仅仅只是介绍 Guava 一小部分功能,还有很对功能,阿粉也觉得很好用在。这里推荐大家去查看 Guava 官方 wiki,查看具体使用方法。

如果大家还想知道其他开源工具类,给阿粉点个赞,下次给大家带来十分好用开源工具类~

相关文章
|
3月前
|
安全 Java 容器
【Java集合类面试二十七】、谈谈CopyOnWriteArrayList的原理
CopyOnWriteArrayList是一种线程安全的ArrayList,通过在写操作时复制新数组来保证线程安全,适用于读多写少的场景,但可能因内存占用和无法保证实时性而有性能问题。
|
3月前
|
存储 安全 Java
Java集合类面试十七】、介绍一下ConcurrentHashMap是怎么实现的?
ConcurrentHashMap在JDK 1.7中通过分段锁实现线程安全,在JDK 1.8中则采用Node数组配合链表和红黑树,并使用Synchronized和CAS操作提高并发性能。
Java集合类面试十七】、介绍一下ConcurrentHashMap是怎么实现的?
|
3月前
|
Java
【Java集合类面试二十一】、请介绍TreeMap的底层原理
TreeMap基于红黑树实现,能够根据键的自然顺序或提供的Comparator排序,其基本操作的时间复杂度为O(log N)。
|
3月前
|
Java
【Java集合类面试二十】、请介绍LinkedHashMap的底层原理
LinkedHashMap的底层原理是在HashMap的基础上,通过维护一条双向链表来保持键值对的插入和遍历顺序,同时继承HashMap的多数方法并重写部分方法以维护链表。
|
3月前
|
Java
【Java集合类面试十九】、说一说你对LinkedHashMap的理解
LinkedHashMap通过维护一个双向链表来保证key-value对的迭代顺序与插入顺序一致,它提供了HashMap的性能和有序迭代的特性,但相比HashMap性能略低。
|
6月前
|
XML Java 测试技术
Java异常处理神器:Guava Throwables类概念与实战
【4月更文挑战第29天】在Java开发中,异常处理是保证程序稳定性和可靠性的关键。Google的Guava库提供了一个强大的工具类Throwables,用于简化和增强异常处理。本篇博客将探讨Throwables类的核心功能及其在实战中的应用。
110 2
|
6月前
|
设计模式 搜索推荐 Java
面试官不按套路出牌,上来就让聊一聊Java中的迭代器(Iterator ),夺命连环问,怎么办?
面试官不按套路出牌,上来就让聊一聊Java中的迭代器(Iterator ),夺命连环问,怎么办?
51 0
|
存储 Java 索引
“速通“ 老生常谈的HashMap [实现原理&&源码解读]
HashMap在现在已然成为了一个老生常谈的话题, 不管是正在学java的小白还是不断跳槽升值的老鸟, 在面试中HashMap几乎不可避免的会被问到, 你可以不被问到, 但你不能不会, 本篇文章的内容就是HashMap的实现原理和源码解读, 各位观众老爷们一起来看看吧
122 1
HashMap? ConcurrentHashMap? 相信看完这篇没人能难住你!(上)
Map 这样的 Key Value 在软件开发中是非常经典的结构,常用于在内存中存放数据。 本篇主要想讨论 ConcurrentHashMap 这样一个并发容器,在正式开始之前我觉得有必要谈谈 HashMap,没有它就不会有后面的 ConcurrentHashMap。
|
安全 Java 容器
HashMap? ConcurrentHashMap? 相信看完这篇没人能难住你!(下)
Map 这样的 Key Value 在软件开发中是非常经典的结构,常用于在内存中存放数据。 本篇主要想讨论 ConcurrentHashMap 这样一个并发容器,在正式开始之前我觉得有必要谈谈 HashMap,没有它就不会有后面的 ConcurrentHashMap。