zoukankan      html  css  js  c++  java
  • 修改Map中确定key对应的value问题

    今天在码代码的时候出现一个没有预料的问题:

    先看下面的代码:

    public static void main(String[] args) {
    		String[] files=new String[]{"abcd","qwer","asdf"};
    		Map<String,Object> map=new HashMap<String,Object>();
    		map.put("file", "12345");
    		map.put("id", 15);
    		map.put("name", "works");
    		
    		List<Map<String,Object>> list=new ArrayList<Map<String,Object>>();
    		for (int i = 0; i < files.length; i++) {
    			list.add(map);
    			list.get(i).put("file", files[i]);
    		}
    		System.out.println(list.toString());		
    	}

    这里map模拟从数据库取到的一条记录,我的本意是根据files的大小生成一个包含n个map的List<Map<String,Object>,list中map的其他key的值都不变,只有key为file的值须要替换为files数组内的内容,于是大致写了上述的处理代码,咋一看上去没有任何问题。运行之后发现,list中的每一个map中key为file的value都为“asdf” !这是什么问题呢,换了多种方法处理,例如,先生成list,再遍历list进行修改,可是结果没有任何变化,超出预期!

    [{id=15, file=asdf, name=works}, {id=15, file=asdf, name=works}, {id=15, file=asdf, name=works}]

    折腾好久,终于发现问题:其实list中的所有元素(map)的引用都是指向内存中的同一块区域,所以上述的修改方式,最终会变成上述输出。

    处理办法:

    public static void main(String[] args) {
    		String[] files=new String[]{"abcd","qwer","asdf"};
    		Map<String,Object> map=new HashMap<String,Object>();
    		map.put("file", "12345");
    		map.put("id", 15);
    		map.put("name", "works");
    		
    		List<Map<String,Object>> list=new ArrayList<Map<String,Object>>();
    		for (int i = 0; i < files.length; i++) {
    			Map<String,Object> hmp=new HashMap<String,Object>();
    			Iterator<String> it=map.keySet().iterator();
    			while (it.hasNext()) {
    				String key = (String) it.next();
    				hmp.put(key, map.get(key));
    			}
    			list.add(hmp);
    			list.get(i).put("file", files[i]);
    		}
    		System.out.println(list.toString());		
    	}

    这样做的目的是每次都new一个Map对象hmp,然后把原来map中的内容复制到新的Map对象中,那么list中的map对象便是分别拥有不同的存储区域。然后对key对应的value进行修改时便不会出现之前被覆盖的问题了。

    [{id=15, file=abcd, name=works}, {id=15, file=qwer, name=works}, {id=15, file=asdf, name=works}]

    出现这个问题,主要是对内存/对象的引用以及Map的相关API理解不深导致

  • 相关阅读:
    使用node.js搭建本地服务器
    使用Vue前端框架实现知乎日报app
    Echarts的使用
    [LeetCode] 642. Design Search Autocomplete System 设计搜索自动补全系统
    [LeetCode] 249. Group Shifted Strings 分组偏移字符串
    [LeetCode] 298. Binary Tree Longest Consecutive Sequence 二叉树最长连续序列
    [LeetCode] 128. Longest Consecutive Sequence 求最长连续序列
    [LeetCode] 227. Basic Calculator II 基本计算器 II
    [LeetCode] 224. Basic Calculator 基本计算器
    [LeetCode] 117. Populating Next Right Pointers in Each Node II 每个节点的右向指针 II
  • 原文地址:https://www.cnblogs.com/elgin-seth/p/5293777.html
Copyright © 2011-2022 走看看