当前位置:   article > 正文

Map封装 (一个键多个值)

map一个键对应多个值

转载:http://android.jobbole.com/83470/

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

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,废话不多说,上代码:

  1. Java
  2. import java.util.List;
  3. import java.util.Map;
  4. import java.util.Set;
  5. /**
  6. * <p>Can save multiple the value of the map.</p>
  7. * Created in Jan 10, 2016 5:00:07 PM.
  8. *
  9. * @author Yan Zhenjie;
  10. */
  11. public interface MultiValueMap<K, V> {
  12. /**
  13. * 添加Key-Value。
  14. *
  15. * @param key key.
  16. * @param value value.
  17. */
  18. void add(K key, V value);
  19. /**
  20. * 添加Key-List<Value>。
  21. *
  22. * @param key key.
  23. * @param values values.
  24. */
  25. void add(K key, List<V> values);
  26. /**
  27. * 设置一个Key-Value,如果这个Key存在就被替换,不存在则被添加。
  28. *
  29. * @param key key.
  30. * @param value values.
  31. */
  32. void set(K key, V value);
  33. /**
  34. * 设置Key-List<Value>,如果这个Key存在就被替换,不存在则被添加。
  35. * @param key key.
  36. * @param values values.
  37. * <a href="http://www.jobbole.com/members/heydee@qq.com">@see</a> #set(Object, Object)
  38. */
  39. void set(K key, List<V> values);
  40. /**
  41. * 替换所有的Key-List<Value>。
  42. *
  43. * @param values values.
  44. */
  45. void set(Map<K, List<V>> values);
  46. /**
  47. * 移除某一个Key,对应的所有值也将被移除。
  48. *
  49. * @param key key.
  50. * @return value.
  51. */
  52. List<V> remove(K key);
  53. /**
  54. * 移除所有的值。
  55. * Remove all key-value.
  56. */
  57. void clear();
  58. /**
  59. * 拿到Key的集合。
  60. * @return Set.
  61. */
  62. Set<K> keySet();
  63. /**
  64. * 拿到所有的值的集合。
  65. *
  66. * @return List.
  67. */
  68. List<V> values();
  69. /**
  70. * 拿到某一个Key下的某一个值。
  71. *
  72. * @param key key.
  73. * @param index index value.
  74. * @return The value.
  75. */
  76. V getValue(K key, int index);
  77. /**
  78. * 拿到某一个Key的所有值。
  79. *
  80. * @param key key.
  81. * @return values.
  82. */
  83. List<V> getValues(K key);
  84. /**
  85. * 拿到MultiValueMap的大小.
  86. *
  87. * @return size.
  88. */
  89. int size();
  90. /**
  91. * 判断MultiValueMap是否为null.
  92. *
  93. * @return True: empty, false: not empty.
  94. */
  95. boolean isEmpty();
  96. /**
  97. * 判断MultiValueMap是否包含某个Key.
  98. *
  99. * @param key key.
  100. * @return True: contain, false: none.
  101. */
  102. boolean containsKey(K key);
  103. }
  104. import java.util.List;
  105. import java.util.Map;
  106. import java.util.Set;
  107. /**
  108. * <p>Can save multiple the value of the map.</p>
  109. * Created in Jan 10, 2016 5:00:07 PM.
  110. *
  111. * @author Yan Zhenjie;
  112. */
  113. public interface MultiValueMap<K, V> {
  114. /**
  115. * 添加Key-Value。
  116. *
  117. * @param key key.
  118. * @param value value.
  119. */
  120. void add(K key, V value);
  121. /**
  122. * 添加Key-List<Value>。
  123. *
  124. * @param key key.
  125. * @param values values.
  126. */
  127. void add(K key, List<V> values);
  128. /**
  129. * 设置一个Key-Value,如果这个Key存在就被替换,不存在则被添加。
  130. *
  131. * @param key key.
  132. * @param value values.
  133. */
  134. void set(K key, V value);
  135. /**
  136. * 设置Key-List<Value>,如果这个Key存在就被替换,不存在则被添加。
  137. * @param key key.
  138. * @param values values.
  139. * <a href="http://www.jobbole.com/members/heydee@qq.com">@see</a> #set(Object, Object)
  140. */
  141. void set(K key, List<V> values);
  142. /**
  143. * 替换所有的Key-List<Value>。
  144. *
  145. * @param values values.
  146. */
  147. void set(Map<K, List<V>> values);
  148. /**
  149. * 移除某一个Key,对应的所有值也将被移除。
  150. *
  151. * @param key key.
  152. * @return value.
  153. */
  154. List<V> remove(K key);
  155. /**
  156. * 移除所有的值。
  157. * Remove all key-value.
  158. */
  159. void clear();
  160. /**
  161. * 拿到Key的集合。
  162. * @return Set.
  163. */
  164. Set<K> keySet();
  165. /**
  166. * 拿到所有的值的集合。
  167. *
  168. * @return List.
  169. */
  170. List<V> values();
  171. /**
  172. * 拿到某一个Key下的某一个值。
  173. *
  174. * @param key key.
  175. * @param index index value.
  176. * @return The value.
  177. */
  178. V getValue(K key, int index);
  179. /**
  180. * 拿到某一个Key的所有值。
  181. *
  182. * @param key key.
  183. * @return values.
  184. */
  185. List<V> getValues(K key);
  186. /**
  187. * 拿到MultiValueMap的大小.
  188. *
  189. * @return size.
  190. */
  191. int size();
  192. /**
  193. * 判断MultiValueMap是否为null.
  194. *
  195. * @return True: empty, false: not empty.
  196. */
  197. boolean isEmpty();
  198. /**
  199. * 判断MultiValueMap是否包含某个Key.
  200. *
  201. * @param key key.
  202. * @return True: contain, false: none.
  203. */
  204. boolean containsKey(K key);
  205. }
  206. 是不是有些小激动呢,接口终于撸完了。是不是就可以用了呢,答案当然是不能,我们知道Map使用的时候假如是用HashMap是这样实例化的:Map<K, V> map = new HashMap<>();,那么我们的MultiValueMap是不是也要提供一个像HashMap一个样实现类呢?
  207. MultiValueMap接口的实现类LinkedMultiValueMap
  208. 实现MultiValueMap接口要implements MultiValueMap...
  209. 既然是可以容纳任何类型的MultiValueMap,实现类LinkedMultiValueMap也要用泛型分析到里我们先把类的雏形写出来:
  210. Java
  211. public class LinkedMultiValueMap<K, V> implements MultiValueMap<K, V> {
  212. ...
  213. }
  214. 1
  215. 2
  216. 3
  217. public class LinkedMultiValueMap<K, V> implements MultiValueMap<K, V> {
  218. ...
  219. }
  220. 看到这里有经验的小伙伴就会发现这尼玛和Map、HashMap一个原理啊,这就对了,当我们看到LinkedMultiValueMap的时候就应该知道我们会用到LinkedHashMap。
  221. 这里把存放数据的原理分析一下,一个Key对应多个Value,我第一个想到的样子是:Map<K, List<Value>>,综上所述我们的数据源就出来了:
  222. Java
  223. public class LinkedMultiValueMap<K, V> implements MultiValueMap<K, V> {
  224. protected Map<K, List<V>> mSource = new LinkedHashMap<K, List<V>>();
  225. ...
  226. }
  227. 1
  228. 2
  229. 3
  230. 4
  231. public class LinkedMultiValueMap<K, V> implements MultiValueMap<K, V> {
  232. protected Map<K, List<V>> mSource = new LinkedHashMap<K, List<V>>();
  233. ...
  234. }
  235. 接下来我们就该实现MultiValueMap接口所有的方法啦,嗯没啥好说的了,上代码,不懂的地方有注释:
  236. Java
  237. import java.util.ArrayList;
  238. import java.util.LinkedHashMap;
  239. import java.util.List;
  240. import java.util.Map;
  241. import java.util.Set;
  242. /**
  243. * Created in Jan 10, 2016 5:03:17 PM.
  244. *
  245. * @author Yan Zhenjie;
  246. */
  247. public class LinkedMultiValueMap<K, V> implements MultiValueMap<K, V> {
  248. protected Map<K, List<V>> mSource = new LinkedHashMap<K, List<V>>();
  249. public LinkedMultiValueMap() {
  250. }
  251. @Override
  252. public void add(K key, V value) {
  253. if (key != null) {
  254. // 如果有这个Key就继续添加Value,没有就创建一个List并添加Value
  255. if (!mSource.containsKey(key))
  256. mSource.put(key, new ArrayList<V>(2));
  257. mSource.get(key).add(value);
  258. }
  259. }
  260. @Override
  261. public void add(K key, List<V> values) {
  262. // 便利添加进来的List的Value,调用上面的add(K, V)方法添加
  263. for (V value : values) {
  264. add(key, value);
  265. }
  266. }
  267. @Override
  268. public void set(K key, V value) {
  269. // 移除这个Key,添加新的Key-Value
  270. mSource.remove(key);
  271. add(key, value);
  272. }
  273. @Override
  274. public void set(K key, List<V> values) {
  275. // 移除Key,添加List<V>
  276. mSource.remove(key);
  277. add(key, values);
  278. }
  279. @Override
  280. public void set(Map<K, List<V>> map) {
  281. // 移除所有值,便利Map里的所有值添加进来
  282. mSource.clear();
  283. mSource.putAll(map);
  284. }
  285. @Override
  286. public List<V> remove(K key) {
  287. return mSource.remove(key);
  288. }
  289. @Override
  290. public void clear() {
  291. mSource.clear();
  292. }
  293. @Override
  294. public Set<K> keySet() {
  295. return mSource.keySet();
  296. }
  297. @Override
  298. public List<V> values() {
  299. // 创建一个临时List保存所有的Value
  300. List<V> allValues = new ArrayList<V>();
  301. // 便利所有的Key的Value添加到临时List
  302. Set<K> keySet = mSource.keySet();
  303. for (K key : keySet) {
  304. allValues.addAll(mSource.get(key));
  305. }
  306. return allValues;
  307. }
  308. @Override
  309. public List<V> getValues(K key) {
  310. return mSource.get(key);
  311. }
  312. @Override
  313. public V getValue(K key, int index) {
  314. List<V> values = mSource.get(key);
  315. if (values != null && index < values.size())
  316. return values.get(index);
  317. return null;
  318. }
  319. @Override
  320. public int size() {
  321. return mSource.size();
  322. }
  323. @Override
  324. public boolean isEmpty() {
  325. return mSource.isEmpty();
  326. }
  327. @Override
  328. public boolean containsKey(K key) {
  329. return mSource.containsKey(key);
  330. }
  331. }
  332. import java.util.ArrayList;
  333. import java.util.LinkedHashMap;
  334. import java.util.List;
  335. import java.util.Map;
  336. import java.util.Set;
  337. /**
  338. * Created in Jan 10, 2016 5:03:17 PM.
  339. *
  340. * @author Yan Zhenjie;
  341. */
  342. public class LinkedMultiValueMap<K, V> implements MultiValueMap<K, V> {
  343. protected Map<K, List<V>> mSource = new LinkedHashMap<K, List<V>>();
  344. public LinkedMultiValueMap() {
  345. }
  346. @Override
  347. public void add(K key, V value) {
  348. if (key != null) {
  349. // 如果有这个Key就继续添加Value,没有就创建一个List并添加Value
  350. if (!mSource.containsKey(key))
  351. mSource.put(key, new ArrayList<V>(2));
  352. mSource.get(key).add(value);
  353. }
  354. }
  355. @Override
  356. public void add(K key, List<V> values) {
  357. // 便利添加进来的List的Value,调用上面的add(K, V)方法添加
  358. for (V value : values) {
  359. add(key, value);
  360. }
  361. }
  362. @Override
  363. public void set(K key, V value) {
  364. // 移除这个Key,添加新的Key-Value
  365. mSource.remove(key);
  366. add(key, value);
  367. }
  368. @Override
  369. public void set(K key, List<V> values) {
  370. // 移除Key,添加List<V>
  371. mSource.remove(key);
  372. add(key, values);
  373. }
  374. @Override
  375. public void set(Map<K, List<V>> map) {
  376. // 移除所有值,便利Map里的所有值添加进来
  377. mSource.clear();
  378. mSource.putAll(map);
  379. }
  380. @Override
  381. public List<V> remove(K key) {
  382. return mSource.remove(key);
  383. }
  384. @Override
  385. public void clear() {
  386. mSource.clear();
  387. }
  388. @Override
  389. public Set<K> keySet() {
  390. return mSource.keySet();
  391. }
  392. @Override
  393. public List<V> values() {
  394. // 创建一个临时List保存所有的Value
  395. List<V> allValues = new ArrayList<V>();
  396. // 便利所有的Key的Value添加到临时List
  397. Set<K> keySet = mSource.keySet();
  398. for (K key : keySet) {
  399. allValues.addAll(mSource.get(key));
  400. }
  401. return allValues;
  402. }
  403. @Override
  404. public List<V> getValues(K key) {
  405. return mSource.get(key);
  406. }
  407. @Override
  408. public V getValue(K key, int index) {
  409. List<V> values = mSource.get(key);
  410. if (values != null && index < values.size())
  411. return values.get(index);
  412. return null;
  413. }
  414. @Override
  415. public int size() {
  416. return mSource.size();
  417. }
  418. @Override
  419. public boolean isEmpty() {
  420. return mSource.isEmpty();
  421. }
  422. @Override
  423. public boolean containsKey(K key) {
  424. return mSource.containsKey(key);
  425. }
  426. }
  427. 到这里,实现类也就完了,高手们已经肯定会用了。
  428. MultiValueMap接口和实现类LinkedMultiValueMap的用法
  429. 其实本来不想写用法的,但是我们还是要本着一个负责人的态度是吧。其实这个封装在开源框架NoHttp中用的很多,很成熟了,下面也再写点用法:
  430. Java
  431. public static void main(String[] args) {
  432. MultiValueMap<String, String> stringMultiValueMap = new LinkedMultiValueMap<>();
  433. // 添加Key为name的
  434. stringMultiValueMap.add("name", "yolanda");
  435. stringMultiValueMap.add("name", "yanzhenjie");
  436. stringMultiValueMap.add("name", "严振杰");
  437. stringMultiValueMap.add("name", "尤兰达");
  438. // 添加Key为domain的
  439. stringMultiValueMap.add("domain", "http://www.yanzhenjie.com");
  440. stringMultiValueMap.add("domain", "http://www.nohttp.net");
  441. // 拿到某个key的某个值
  442. System.out.println("name的第三个值:" + stringMultiValueMap.getValue("name", 3));
  443. // 打印所有值
  444. Set<String> keySet = stringMultiValueMap.keySet();
  445. for (String key : keySet) {
  446. List<String> values = stringMultiValueMap.getValues(key);
  447. for (String value : values) {
  448. System.out.println(key + ": " + value);
  449. }
  450. }
  451. }
  452. public static void main(String[] args) {
  453. MultiValueMap<String, String> stringMultiValueMap = new LinkedMultiValueMap<>();
  454. // 添加Key为name的
  455. stringMultiValueMap.add("name", "yolanda");
  456. stringMultiValueMap.add("name", "yanzhenjie");
  457. stringMultiValueMap.add("name", "严振杰");
  458. stringMultiValueMap.add("name", "尤兰达");
  459. // 添加Key为domain的
  460. stringMultiValueMap.add("domain", "http://www.yanzhenjie.com");
  461. stringMultiValueMap.add("domain", "http://www.nohttp.net");
  462. // 拿到某个key的某个值
  463. System.out.println("name的第三个值:" + stringMultiValueMap.getValue("name", 3));
  464. // 打印所有值
  465. Set<String> keySet = stringMultiValueMap.keySet();
  466. for (String key : keySet) {
  467. List<String> values = stringMultiValueMap.getValues(key);
  468. for (String value : values) {
  469. System.out.println(key + ": " + value);
  470. }
  471. }
  472. }
  473. 我们run一下看结果:
  474. Java
  475. name的第三个值:尤兰达
  476. name: yolanda
  477. name: yanzhenjie
  478. name: 严振杰
  479. name: 尤兰达
  480. domain: http://www.yanzhenjie.com
  481. domain: http://www.nohttp.net
  482. name的第三个值:尤兰达
  483. name: yolanda
  484. name: yanzhenjie
  485. name: 严振杰
  486. name: 尤兰达
  487. domain: http://www.yanzhenjie.com
  488. domain: http://www.nohttp.net

转载:http://android.jobbole.com/83470/

转载于:https://www.cnblogs.com/uniquezhangqi/p/9199450.html

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/空白诗007/article/detail/991955
推荐阅读
相关标签
  

闽ICP备14008679号