NoHttp开源地址:https://github.com/yanzhenjie/NoHttp
NoHttp详细使用文档已发布,你想知道的全都有,请点我移步!
版权声明:转载请注明本文转自严振杰的博客: http://blog.yanzhenjie.com
好久没有写博客啦,今天来买个萌。
在日常的开发中经常用到键值对,也就是Map
啦,我们知道Map是一个接口,它的特点是一个Key对应一个Value,也就是一个键对应一个值,但是往往我们需要一个Key对应多个Value,这种时候系统的API就无法满足我们的需要了,因此今天介绍一个我封装的一个Key对应多个Value,既一个键对应多个值的MultiValueMap
。
第二个,我们的MultiValueMap
要想什么Key-Value都可以放,就要用到泛型,泛型不好的同学也不用担心,只要你用过Map
、HashMap
我保准你看得懂。
用过Map
的人都知道,Map
是一个接口(interface),所以我们这里也把MultiValueMap
定义成一个接口。那么需要那些方法呢,我们来列个表:
- 添加一个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,废话不多说,上代码:
import java.util.List;
import java.util.Map;
import java.util.Set;
/**
* Can save multiple the value of the map.
* 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。
*
* @param key key.
* @param values values.
*/
void add(K key, List values);
/**
* 设置一个Key-Value,如果这个Key存在就被替换,不存在则被添加。
*
* @param key key.
* @param value values.
*/
void set(K key, V value);
/**
* 设置Key-List,如果这个Key存在就被替换,不存在则被添加。
* @param key key.
* @param values values.
* @see #set(Object, Object)
*/
void set(K key, List values);
/**
* 替换所有的Key-List。
*
* @param values values.
*/
void set(Map> values);
/**
* 移除某一个Key,对应的所有值也将被移除。
*
* @param key key.
* @return value.
*/
List remove(K key);
/**
* 移除所有的值。
* Remove all key-value.
*/
void clear();
/**
* 拿到Key的集合。
* @return Set.
*/
Set keySet();
/**
* 拿到所有的值的集合。
*
* @return List.
*/
List 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 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
,那么我们的MultiValueMap
是不是也要提供一个像HashMap
一个样实现类呢?
MultiValueMap
接口要implements MultiValueMap...
既然是可以容纳任何类型的MultiValueMap
,实现类LinkedMultiValueMap
也要用泛型
分析到里我们先把类的雏形写出来:
public class LinkedMultiValueMap<K, V> implements MultiValueMap<K, V> {
...
}
看到这里有经验的小伙伴就会发现这尼玛和Map
、HashMap
一个原理啊,这就对了,当我们看到LinkedMultiValueMap
的时候就应该知道我们会用到LinkedHashMap
。
这里把存放数据的原理分析一下,一个Key对应多个Value,我第一个想到的样子是:Map
,综上所述我们的数据源就出来了:
public class LinkedMultiValueMap<K, V> implements MultiValueMap<K, V> {
protected Map> mSource = new LinkedHashMap>();
...
}
接下来我们就该实现MultiValueMap
接口所有的方法啦,嗯没啥好说的了,上代码,不懂的地方有注释:
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> mSource = new LinkedHashMap>();
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(2));
mSource.get(key).add(value);
}
}
@Override
public void add(K key, List 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 values) {
// 移除Key,添加List
mSource.remove(key);
add(key, values);
}
@Override
public void set(Map> map) {
// 移除所有值,便利Map里的所有值添加进来
mSource.clear();
mSource.putAll(map);
}
@Override
public List remove(K key) {
return mSource.remove(key);
}
@Override
public void clear() {
mSource.clear();
}
@Override
public Set keySet() {
return mSource.keySet();
}
@Override
public List values() {
// 创建一个临时List保存所有的Value
List allValues = new ArrayList();
// 便利所有的Key的Value添加到临时List
Set keySet = mSource.keySet();
for (K key : keySet) {
allValues.addAll(mSource.get(key));
}
return allValues;
}
@Override
public List getValues(K key) {
return mSource.get(key);
}
@Override
public V getValue(K key, int index) {
List 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);
}
}
到这里,实现类也就完了,高手们已经肯定会用了。
其实本来不想写用法的,但是我们还是要本着一个负责人的态度是吧。其实这个封装在开源框架NoHttp中用的很多,很成熟了,下面也再写点用法:
public static void main(String[] args) {
MultiValueMap 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 keySet = stringMultiValueMap.keySet();
for (String key : keySet) {
List 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