关闭

Map之一个Key存多个Value的MultiValueMap(一个键多个值)

标签: JavaMapAndroid键值对一个键多个值
10279人阅读 评论(5) 收藏 举报
分类:

NoHttp开源地址:https://github.com/yanzhenjie/NoHttp

NoHttp详细使用文档已发布,你想知道的全都有,请点我移步!

版权声明:转载请注明本文转自严振杰的博客: http://blog.yanzhenjie.com


前言

  好久没有写博客啦,今天来买个萌。
  在日常的开发中经常用到键值对,也就是Map啦,我们知道Map是一个接口,它的特点是一个Key对应一个Value,也就是一个键对应一个值,但是往往我们需要一个Key对应多个Value,这种时候系统的API就无法满足我们的需要了,因此今天介绍一个我封装的一个Key对应多个Value,既一个键对应多个值的MultiValueMap
  第二个,我们的MultiValueMap要想什么Key-Value都可以放,就要用到泛型,泛型不好的同学也不用担心,只要你用过MapHashMap我保准你看得懂。

MultiValueMap接口的定义

  用过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;

/**
 * <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一个样实现类呢?

MultiValueMap接口的实现类LinkedMultiValueMap

  1. 实现MultiValueMap接口要implements MultiValueMap...
  2. 既然是可以容纳任何类型的MultiValueMap,实现类LinkedMultiValueMap也要用泛型

      分析到里我们先把类的雏形写出来:

public class LinkedMultiValueMap<K, V> implements MultiValueMap<K, V> {
    ...
}

  看到这里有经验的小伙伴就会发现这尼玛和MapHashMap一个原理啊,这就对了,当我们看到LinkedMultiValueMap的时候就应该知道我们会用到LinkedHashMap
  这里把存放数据的原理分析一下,一个Key对应多个Value,我第一个想到的样子是:Map<K, List<Value>>,综上所述我们的数据源就出来了:

public class LinkedMultiValueMap<K, V> implements MultiValueMap<K, V> {
    protected Map<K, List<V>> mSource = new LinkedHashMap<K, List<V>>();
    ...
}

  接下来我们就该实现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<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);
    }

}

  到这里,实现类也就完了,高手们已经肯定会用了。

MultiValueMap接口和实现类LinkedMultiValueMap的用法

  其实本来不想写用法的,但是我们还是要本着一个负责人的态度是吧。其实这个封装在开源框架NoHttp中用的很多,很成熟了,下面也再写点用法:

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
  

7
0
查看评论
发表评论
* 以上用户言论只代表其个人观点,不代表CSDN网站的观点或立场

java JsonObject的用法以及map总同一个key,但是不同的value,如何存储的问题

最近在迁移一份代码的时候,遇到一些问题。   要解决的问题如下:   需要把 days , type , list 封装为一个JSONObject对象,然后传给另一个web项目,在另一个web项目中再...
  • u011415481
  • u011415481
  • 2015-12-29 18:33
  • 5325

Redis实现键对应多值

Redis实现键对应多值作者:chszs,未经博主允许不得转载。经许可的转载需注明作者和博客主页:http://blog.csdn.net/chszsRedis服务器提供了很多流行的数据结构,比如Ma...
  • chszs
  • chszs
  • 2016-03-14 21:25
  • 6866

多个KEY值对应一个VALUE的MAP

http://www.deuxmille.org/archives/1797 在整理国内期货的高频数据,也就是CTP里面发送的500毫秒级的交易信息。目前有几个数据源,首要任务就是要把几个数据源校验...
  • l799623787
  • l799623787
  • 2013-03-21 10:16
  • 1684

在Java中自定义的一个key对多个Value的map

实现类: import java.util.*; /** * 自定义一对多map */ public class OToMoreMap{ private List mkey; ...
  • cheng9981
  • cheng9981
  • 2017-01-09 22:06
  • 1779

java Map 一个key其实可以保存多个value

我们平时使用的Map,都是只能在Map中保存一个相同的Key,我们后面保存的相同的key都会将原来的key的值覆盖掉,如下面的例子。 package test62; import java.util...
  • wx_962464
  • wx_962464
  • 2012-06-29 00:42
  • 30432

java 中实现一个key保存多个value

转自:  http://blog.csdn.net/wx_962464/article/details/7701141 我们平时使用的HashMap,都是只能在Map中保存一个相同的Key,...
  • Manketon
  • Manketon
  • 2015-04-06 17:28
  • 324

shell模拟php多进程从redis获取数据(一个库多个key值)

背景:现在的数据已经写到了redis队列里面,完成了入栈的操作,后期打算从redis获取数据,完成出栈的操作,出栈后然后做一系列的逻辑处理       环境: VMware虚拟机  内存:1G ...
  • gb4215287
  • gb4215287
  • 昨天 18:52
  • 5

multimap中一个key对应多个键值的查询处理

From:http://zhidao.baidu.com/link?url=s-lktjYYv8TBzqZonvQn0SxGLf7Vgk252-5bNUUxx9wsL_vonx57HvFeN4KN06...
  • u013427969
  • u013427969
  • 2016-09-27 00:04
  • 1343

编写一个类,在main方法中定义一个Map对象(采用泛型),加入若干个对象,然后遍历并打印出各元素的key和value。

编写一个类,在main方法中定义一个package com.itheima; import java.util.Set; import java.util.TreeMap; /** *第3题:编...
  • yz972641975
  • yz972641975
  • 2015-08-21 17:03
  • 2325

找出一个数组中重复次数最多的字符暨找出Map中的最大Value及其对应的Key

rt:找出Map中的最大Value及其对应的Key,常用于找出一个数组中重复次数最多的字符import java.util.*; public class KeyOnMaxValue{ public...
  • qq_30629571
  • qq_30629571
  • 2016-06-15 21:19
  • 439
    个人资料
    • 访问:1607957次
    • 积分:6525
    • 等级:
    • 排名:第4287名
    • 原创:52篇
    • 转载:0篇
    • 译文:1篇
    • 评论:1274条
    我的微信公众号
    欢迎关注我的公众号,不定期为您推送优选博文,生活趣事!
    关注我的微信公众号

    关注我的微博
    友情链接


    我的Github

    QQ交流群 547839514

    博客专栏