• 修改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理解不深导致

  • 相关阅读:
    Python reportlab table 设置cellstyle枚举,设置单元格padding
    代理工具 v2ra*的使用
    python 开发环境需要安装
    postgres 新增或者更新语句
    python psycopg2 查询结果返回字典类型
    postgres 多个字段匹配查询
    django 执行原生sql形参传递,字段参数传递
    亚马逊接口调用
    python x 开头的字符转码为中文
    postgres 定义变量
  • 原文地址:https://www.cnblogs.com/elgin-seth/p/5293777.html
Copyright © 2020-2023  润新知