在平常的开发中常常用到键值对,也就是Map
啦,咱们知道Map是一个接口,它的特色是一个Key对应一个Value,也就是一个键对应一个值,可是每每咱们须要一个Key对应多个Value,这种时候系统的API就没法知足咱们的须要了,所以今天介绍一个我封装的一个Key对应多个Value,既一个键对应多个值的MultiValueMap
。
第二个,咱们的MultiValueMap
要想什么Key-Value均可以放,就要用到泛型,泛型很差的同窗也不用担忧,只要你用过Map
、HashMap
我保准你看得懂。java
用过Map
的人都知道,Map
是一个接口(interface),因此咱们这里也把MultiValueMap
定义成一个接口。那么须要那些方法呢,咱们来列个表:git
- 添加一个Key对应一个Value的:void add(K, V);
- 添加一个Key对应多个Value的:void add(K, List);
- 设置一个Key对应一个Value的:void set(K, V);
- 设置一个Key对应多个Value的:void set(K, List);
- 移除一个Key和它对应的Value:List remove(K);
- 清除全部Value的:void clear();
- 拿到全部Key的集合:Set keySet();
- 拿到全部的值:List values();
- 拿到一个Key对应的某个值:V getValue(K, index);
- 拿到一个Key对应的全部Value:List getValues(K);
- 拿到MultiValueMap的大小:int size();
- 判断是否为空MultiValueMap:boolean isEmpty();
- 判断是否包含某个Key:boolean containKey(K);
下面咱们根据分析先把MultiValueMap
接口定义出来,其实这个封装是个人开源项目NoHttp中提出来的,有兴趣的看官光顾下NoHttp,废话很少说,上代码:github
import java.util.List; import java.util.Map; import java.util.Set; /** * <p>Can save multiple the value of the map.</p> * Created in Jan 10, 2016 5:00:07 PM. * * @author Yan Zhenjie; */ public interface MultiValueMap<K, V> { /** * 添加Key-Value。 * * @param key key. * @param value value. */ void add(K key, V value); /** * 添加Key-List<Value>。 * * @param key key. * @param values values. */ void add(K key, List<V> values); /** * 设置一个Key-Value,若是这个Key存在就被替换,不存在则被添加。 * * @param key key. * @param value values. */ void set(K key, V value); /** * 设置Key-List<Value>,若是这个Key存在就被替换,不存在则被添加。 * @param key key. * @param values values. * @see #set(Object, Object) */ void set(K key, List<V> values); /** * 替换全部的Key-List<Value>。 * * @param values values. */ void set(Map<K, List<V>> values); /** * 移除某一个Key,对应的全部值也将被移除。 * * @param key key. * @return value. */ List<V> remove(K key); /** * 移除全部的值。 * Remove all key-value. */ void clear(); /** * 拿到Key的集合。 * @return Set. */ Set<K> keySet(); /** * 拿到全部的值的集合。 * * @return List. */ List<V> values(); /** * 拿到某一个Key下的某一个值。 * * @param key key. * @param index index value. * @return The value. */ V getValue(K key, int index); /** * 拿到某一个Key的全部值。 * * @param key key. * @return values. */ List<V> getValues(K key); /** * 拿到MultiValueMap的大小. * * @return size. */ int size(); /** * 判断MultiValueMap是否为null. * * @return True: empty, false: not empty. */ boolean isEmpty(); /** * 判断MultiValueMap是否包含某个Key. * * @param key key. * @return True: contain, false: none. */ boolean containsKey(K key); }
是否是有些小激动呢,接口终于撸完了。是否是就能够用了呢,答案固然是不能,咱们知道Map
使用的时候假如是用HashMap
是这样实例化的:Map<K, V> map = new HashMap<>();
,那么咱们的MultiValueMap
是否是也要提供一个像HashMap
一个样实现类呢?markdown
MultiValueMap
接口要implements MultiValueMap...
既然是能够容纳任何类型的MultiValueMap
,实现类LinkedMultiValueMap
也要用泛型网络
分析到里咱们先把类的雏形写出来:框架
public class LinkedMultiValueMap<K, V> implements MultiValueMap<K, V> { ... }
看到这里有经验的小伙伴就会发现这尼玛和Map
、HashMap
一个原理啊,这就对了,当咱们看到LinkedMultiValueMap
的时候就应该知道咱们会用到LinkedHashMap
。
这里把存放数据的原理分析一下,一个Key对应多个Value,我第一个想到的样子是:Map<K, List<Value>>
,综上所述咱们的数据源就出来了:dom
public class LinkedMultiValueMap<K, V> implements MultiValueMap<K, V> { protected Map<K, List<V>> mSource = new LinkedHashMap<K, List<V>>(); ... }
接下来咱们就该实现MultiValueMap
接口全部的方法啦,嗯没啥好说的了,上代码,不懂的地方有注释:ide
import java.util.ArrayList; import java.util.LinkedHashMap; import java.util.List; import java.util.Map; import java.util.Set; /** * Created in Jan 10, 2016 5:03:17 PM. * * @author Yan Zhenjie; */ public class LinkedMultiValueMap<K, V> implements MultiValueMap<K, V> { protected Map<K, List<V>> mSource = new LinkedHashMap<K, List<V>>(); public LinkedMultiValueMap() { } @Override public void add(K key, V value) { if (key != null) { // 若是有这个Key就继续添加Value,没有就建立一个List并添加Value if (!mSource.containsKey(key)) mSource.put(key, new ArrayList<V>(2)); mSource.get(key).add(value); } } @Override public void add(K key, List<V> values) { // 便利添加进来的List的Value,调用上面的add(K, V)方法添加 for (V value : values) { add(key, value); } } @Override public void set(K key, V value) { // 移除这个Key,添加新的Key-Value mSource.remove(key); add(key, value); } @Override public void set(K key, List<V> values) { // 移除Key,添加List<V> mSource.remove(key); add(key, values); } @Override public void set(Map<K, List<V>> map) { // 移除全部值,便利Map里的全部值添加进来 mSource.clear(); mSource.putAll(map); } @Override public List<V> remove(K key) { return mSource.remove(key); } @Override public void clear() { mSource.clear(); } @Override public Set<K> keySet() { return mSource.keySet(); } @Override public List<V> values() { // 建立一个临时List保存全部的Value List<V> allValues = new ArrayList<V>(); // 便利全部的Key的Value添加到临时List Set<K> keySet = mSource.keySet(); for (K key : keySet) { allValues.addAll(mSource.get(key)); } return allValues; } @Override public List<V> getValues(K key) { return mSource.get(key); } @Override public V getValue(K key, int index) { List<V> values = mSource.get(key); if (values != null && index < values.size()) return values.get(index); return null; } @Override public int size() { return mSource.size(); } @Override public boolean isEmpty() { return mSource.isEmpty(); } @Override public boolean containsKey(K key) { return mSource.containsKey(key); } }
到这里,实现类也就完了,高手们已经确定会用了。post
其实原本不想写用法的,可是咱们仍是要本着一个负责人的态度是吧。其实这个封装在开源框架NoHttp中用的不少,很成熟了,下面也再写点用法:spa
public static void main(String[] args) { MultiValueMap<String, String> stringMultiValueMap = new LinkedMultiValueMap<>(); // 添加Key为name的 stringMultiValueMap.add("name", "yolanda"); stringMultiValueMap.add("name", "yanzhenjie"); stringMultiValueMap.add("name", "严振杰"); stringMultiValueMap.add("name", "尤兰达"); // 添加Key为domain的 stringMultiValueMap.add("domain", "http://www.yanzhenjie.com"); stringMultiValueMap.add("domain", "http://www.nohttp.net"); // 拿到某个key的某个值 System.out.println("name的第三个值:" + stringMultiValueMap.getValue("name", 3)); // 打印全部值 Set<String> keySet = stringMultiValueMap.keySet(); for (String key : keySet) { List<String> values = stringMultiValueMap.getValues(key); for (String value : values) { System.out.println(key + ": " + value); } } }
咱们run一下看结果:
name的第三个值:尤兰达
name: yolanda
name: yanzhenjie
name: 严振杰
name: 尤兰达
domain: http://www.yanzhenjie.com domain: http://www.nohttp.net
果真没有问题呀,我得意的笑呀哈哈哈……
仍是有疑问的同窗在下面留言噢,天很晚了,睡觉。
欢迎你们关注Android开源网络框架NoHttp:https://github.com/yanzhenjie/NoHttp
在线直播视频和代码下载:http://pan.baidu.com/s/1miEOtwG