MultiMap
类关系
ArrayListMultiMap.java
Multimap <I>
|
|
AbstractMultimap <A> Serializable <I>
|__________________________________________|
|
AbstractMapBasedMultimap <A>
|
|
AbstractListMultimap <A> ListMultimap <I>
|__________________________________________|
|
ArrayListMultiMap
LinkedListMultiMap.java
Multimap <I>
|
|
AbstractMultimap <A> ListMultimap <I> Serializable <I>
|__________________________________________|______________________________|
|
LinkedListMultimap
TreeMultimap.java
Multimap <I>
|
|
AbstractMultimap <A> Serializable <I>
|__________________________________________|
|
AbstractMapBasedMultimap <A> SetMultimap <I>
|__________________________________________|
|
AbstractSetMultimap <A> SortedSetMultimap <I>
|__________________________________________|
|
AbstractSortedSetMultimap <A>
|
|
AbstractSortedKeySortedSetMultimap <A>
|
|
TreeMultimap
实现方法
ArrayListMultimap
ArrayListMultiMap.java是以ArrayList为Collection的特定实现,这个类中没有太多的实际代码,主要是createCollection()方法中特定的产生一个ArrayList作为Collection。
AbstractListMultimap.java是AbstractMultimap的一个List专有版本,这个类和ListMultimap接口一起,将Multimap的方法都重写为List。
AbstractMapBasedMultimap.java则是所有以Map为核心的Multimap的基本实现,这里实现了所有Multimap的方法,是最重要的一部分。
clear方法,先将每个collection清空,再把map清空
public void clear() {
// Clear each collection, to make previously returned collections empty.
for (Collection<V> collection : map.values()) {
collection.clear();
}
map.clear();
totalSize = 0;
}
put方法,我们可以发现这里的size是每多一个KV对就加1,而不是唯一Key的数量,这点和Map不同
public boolean put(@Nullable K key, @Nullable V value) {
Collection<V> collection = map.get(key);
// 如果这是一个新key没有对应的Collection
if (collection == null) {
// 先根据子类实现创建一个相应的Collection
collection = createCollection(key);
// 将KV对加入Map中
if (collection.add(value)) {
totalSize++;
map.put(key, collection);
return true;
} else {
throw new AssertionError("New Collection violated the Collection spec");
}
// 如果已经有这个key了,就加入它的Collection
} else if (collection.add(value)) {
totalSize++;
return true;
} else {
return false;
}
}
removeAll方法,需要返回删除的Value,所以要一个临时变量存起来
public Collection<V> removeAll(@Nullable Object key) {
// 先将Key移出
Collection<V> collection = map.remove(key);
// 如果Value为空,则返回空集合
if (collection == null) {
return createUnmodifiableEmptyCollection();
}
// 否则将Value的值拷贝到输出集合中,再把Value清空
Collection<V> output = createCollection();
output.addAll(collection);
totalSize -= collection.size();
collection.clear();
// 返回输出集合
return unmodifiableCollectionSubclass(output);
}
size方法,注意这里返回的是KV数,而非K的唯一个数
public int size() {
return totalSize;
}
isEmpty方法,判断KV数是否为0
public boolean isEmpty() {
return size() == 0;
}
get方法
public Collection<V> get(@Nullable K key) {
Collection<V> collection = map.get(key);
// 如果没有Value就新建一个Collection
if (collection == null) {
collection = createCollection(key);
}
return wrapCollection(key, collection);
}
AbstractMultimap.java也实现了一些基本方法,和上一个文件一起涵盖了所有API
remove方法
public boolean remove(@Nullable Object key, @Nullable Object value) {
// 从Map中按照key找到该集合
Collection<V> collection = asMap().get(key);
// 如果集合不为空,则找到集合中的这个值并删除
return collection != null && collection.remove(value);
}
putAll方法,以key为第一参数,以一个Iterable为第二参数
public boolean putAll(@Nullable K key, Iterable<? extends V> values) {
checkNotNull(values);
// make sure we only call values.iterator() once
// and we only call get(key) if values is nonempty
if (values instanceof Collection) {
Collection<? extends V> valueCollection = (Collection<? extends V>) values;
// 拿出Collection然后把新的addAll进去
return !valueCollection.isEmpty() && get(key).addAll(valueCollection);
} else {
Iterator<? extends V> valueItr = values.iterator();
// 拿出Iterator然后把新的addAll进去
return valueItr.hasNext() && Iterators.addAll(get(key), valueItr);
}
}
putAll方法,以另一个Multimap为参数
public boolean putAll(Multimap<? extends K, ? extends V> multimap) {
boolean changed = false;
for (Map.Entry<? extends K, ? extends V> entry : multimap.entries()) {
changed |= put(entry.getKey(), entry.getValue());
}
return changed;
}
replace方法
public Collection<V> replaceValues(@Nullable K key, Iterable<? extends V> values) {
checkNotNull(values);
// 先移除所有
Collection<V> result = removeAll(key);
// 在加入新的值
putAll(key, values);
return result;
}
containsValue方法
public boolean containsValue(@Nullable Object value) {
// 检查每一个Collection是否有这个值
for (Collection<V> collection : asMap().values()) {
if (collection.contains(value)) {
return true;
}
}
return false;
}
containsEntry方法,实际上是用get来判断是否有这个key,然后再看是否有这个值
public boolean containsEntry(@Nullable Object key, @Nullable Object value) {
Collection<V> collection = asMap().get(key);
return collection != null && collection.contains(value);
}
MultiMap的更多相关文章
- C++ std::multimap
std::multimap template < class Key, // multimap::key_type class T, // multimap::mapped_type class ...
- Guava学习笔记:Guava新增集合类型-Multimap
在日常的开发工作中,我们有的时候需要构造像Map<K, List<V>>或者Map<K, Set<V>>这样比较复杂的集合类型的数据结构,以便做相应的业 ...
- [Google Guava]学习--新集合类型Multimap
每个有经验的Java程序员都在某处实现过Map<K, List<V>>或Map<K, Set<V>>,并且要忍受这个结构的笨拙. 假如目前有个需求是给两 ...
- 4.3 map和multimap
使用map multimap必须包含头文件map *:multimap 1)multimap定义 template<class Key,class Pred=less<Key>,cl ...
- STL中的map/multimap小结
(1)使用map/multimap之前必须包含头文件<map>:#include<map> 并且和所有的关联式容器一样,map/multimap通常以平衡二叉树来完成 (2)n ...
- 问题解决——multimap中统计key的种类
================声明================= 本文原创,转载请注明出处和作者,并保持文章的完整性. 本文链接:http://www.cnblogs.com/wlsandwho ...
- STL之multimap
参见http://www.cplusplus.com/reference/map/multimap/ 多重映射multimap和map映射很相似,但是multimap允许重复的关键字,这使得multi ...
- 使用multimap创建重复键关联容器
在“使用 <map> 库创建关联容器”一文中,我们讨论了标准库中的 map 关联容器.但那只是 map 容器的一部分.标准库还定义了一个 multimap 容器,它与 map 类似,所不同 ...
- STL学习系列九:Map和multimap容器
1.map/multimap的简介 map是标准的关联式容器,一个map是一个键值对序列,即(key,value)对.它提供基于key的快速检索能力. map中key值是唯一的.集合中的元素按一定的顺 ...
随机推荐
- Python概述_软件安装_常见问题
1. Python安装 目前python有2个大版本,2和3,由于2和3语法有差别,现有的许多库都是基于python2开发,本系列文章以python2为主. 1.1 重要概念 1. 动态语言 运行 ...
- Android核心分析之二十八Android GDI之Surface&Canvas
Surface&Canvas Canvas为在画布的意思.Android上层的作图几乎都通过Canvas实例来完成,其实Canvas更多是一种接口的包装.drawPaints ,drawPoi ...
- 【转】SIP初步
1.什么是SIP SIP(会话发起协议)属于IP应用层协议,用于在IP网上为用户提供会话应用.会话(Session)指两方或多方用户之间的语音.视频.及其他媒体形式的通信,具体可能是IP电话.会议.即 ...
- 五分钟轻松搞定产品需求文档!这可能史上最全PRD文档模板
本文由 @JustWu 原创发布于社区 为什么写这篇文章? 第一:写PMCAFF的PRD文档,大家都是用户,比较好参考与理解,方便大家来找我写的不好的地方. 第二:我在自学PRD文档的编写过程中,总 ...
- QWidget的六个刷新函数(居然有QWidget::erase函数,且并不产生绘制事件)
Qt paintevent事件 一.主要理解一下几个方法和属性: 1.QWidget * QScrollView::viewport () const 2.void QWidget::paintE ...
- WaitForSingleObject与WaitForMultipleObjects用法详解(好用,而且进入一个非常高效沉睡状态,只占用极少的CPU时间片)
在多线程下面,有时候会希望等待某一线程完成了再继续做其他事情,要实现这个目的,可以使用Windows API函数WaitForSingleObject,或者WaitForMultipleObjects ...
- iOS iPhone iPad 各种控件默认高度
iPhone iPad 各种控件默认高度 注意:这些是ios7之前的,ios7之后(包括ios7)有改动,我会在后面标注出来 iPhone和iPad下各种常见控件的宽度和标准是一样的,所以这里就用iP ...
- AppDomain 应用程序域
应用程序域 一.什么是应用程序域? 应用程序域 (application domain) (AppDomain) 一种边界,它由公共语言运行库围绕同一应用程序范围内创建的对象建立(即,从应用程序入口点 ...
- linux 配置 wlan 连接
第一步,先确定你已经安装了无线网卡驱动.我的是 ath9k ,linux 内核不自带,现编译成模块 然后安装上.具体步骤请自行搜索 linux 编译内核 第二步,起用模块 modprobe ath9k ...
- sql2000无法打开1433端口及解决方法
1.如果你是win2003,那么一定要安装sql的补丁sp3a以上版本SP 检查你的SQL有没有打补丁,没有的话要打上补丁,检查的方法是在查询分析器中运行:select @@version如果出来的版 ...