这属于我在开发中碰过的坑 ,容器中存放者对象,当clear()的时候,出现的奇葩问题。好了,直接看代码:

package com.DataType.yinyong;

import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map; public class TestMethod {
//我总结了一下,因为实际开发中涉及到代码层级较深,所以有时候会不容易看出问题
public static void main(String[] args) {
TestMethod testMethod =new TestMethod();
List<Map<String, List<Object>>> resultMaps = new ArrayList<Map<String, List<Object>>>();
resultMaps.add(testMethod.getList());
testMethod.testXunHuan(resultMaps); } public void testXunHuan(List<Map<String, List<Object>>> resultMaps) {
for (Map<String, List<Object>> resultMap : resultMaps) {
int resultNum = 0;
List<Object> result = resultMap.get("r1");
resultMap.remove("r1");
if (result == null)
continue;
resultNum = result.size();
if (resultNum <= 0) {
continue;
}
for (int i = 0; i < result.size(); i++) {
Object o = result.get(i);
}
result.clear();
System.out.println("虽然 我修改的result是指向r1,但是我的map内的另外一个值 :"+resultMap.get("r12"));
}
} public Map<String, List<Object>> getList() {
Map<String, List<Object>> resultMap = new HashMap<String, List<Object>>();
//下面我提供一个List<Object>,并且让resultMap中不同的key引用相同的值
Map<String, List<Object>> srcMap = new HashMap<String, List<Object>>();
List<Object> srcList = new ArrayList<Object>();
for(int i = 0 ; i < 10;i++){
srcList.add("count");
}
srcMap.put("key1", srcList);
srcMap.put("key2", srcList);
//下面我声明一个新的list用来
List<Object> result = srcMap.get("key1");
resultMap.put("r1", result);
List<Object> result2 = srcMap.get("key2");
resultMap.put("r12", result2);
System.out.println("resul实际上是一个"+(result ==result2) );
return resultMap;
} }

输出结果:

总结 :

  1,java中是弱化了引用还是传值的概念,也不需要考虑指针。

    2, 但有些时候,特别是容器内还是需要注意一下,有可能Map中两个key指向的value是同一个对象,这时候,当你选择拿到一个值 clear(),就会出现错误

  3, 于是你可以采用两种方式解决这一问题

     ① :如果不去clear(),的确不会对其他数据产生影响,但是会造成内存消耗大,实际开发不可取(实际开发中不可能只有这两个键值对)

     ② :可以选择copy一下数组,但是同样的原因导致不可取(注意collections.copy()方法是如何使用的

public Map<String, List<Object>> getList() {
Map<String, List<Object>> resultMap = new HashMap<String, List<Object>>();
//下面我提供一个List<Object>,并且让resultMap中不同的key引用相同的值
Map<String, List<Object>> srcMap = new HashMap<String, List<Object>>();
List<Object> srcList = new ArrayList<Object>();
for(int i = 0 ; i < 10;i++){
srcList.add("count");
}
List<Object> srcList1 = new ArrayList<Object>();
Collections.addAll(srcList1, new Object[srcList.size()]);
Collections.copy(srcList1, srcList);//collection.copy(dest,src)方法需要dest的size大于等于src、属于深copy
srcMap.put("key1", srcList);
srcMap.put("key2", srcList1);
//下面我声明一个新的list用来
List<Object> result = srcMap.get("key1");
resultMap.put("r1", result);
List<Object> result2 = srcMap.get("key2");
resultMap.put("r12", result2);
System.out.println("resul实际上是一个List么"+(result ==result2) );
return resultMap;
}

  4;最终我决定,每次clear之前,先去判断是否存在有其他key用到

public void testXunHuan(List<Map<String, List<Object>>> resultMaps) {
for (Map<String, List<Object>> resultMap : resultMaps) {
int resultNum = 0;
List<Object> result = resultMap.get("r1");
resultMap.remove("r1");
if (result == null)
continue;
resultNum = result.size();
if (resultNum <= 0) {
continue;
}
for (int i = 0; i < result.size(); i++) {
Object o = result.get(i);
}
boolean isExistSame =false;
Iterator< List<Object>> ite= resultMap.values().iterator();
while(ite.hasNext()){
if(ite.next().equals(result)){
isExistSame = true;
break;
}
}
//只有不存在相同值,才可以删除
if(!isExistSame){
result.clear();
}
System.out.println("虽然 我修改的result是指向r1,但是我的map内的另外一个值 :"+resultMap.get("r12"));
}
}

参考:

由java中深度复制一伸出Collections.copy的使用

Java中的深拷贝(深复制)和浅拷贝(浅复制)

渐析java的浅拷贝和深拷贝

    

Java开发中碰到的Map的坑的更多相关文章

  1. 编写高质量代码:改善Java程序的151个建议(第一章:JAVA开发中通用的方法和准则)

    编写高质量代码:改善Java程序的151个建议(第一章:JAVA开发中通用的方法和准则) 目录 建议1: 不要在常量和变量中出现易混淆的字母 建议2: 莫让常量蜕变成变量 建议3: 三元操作符的类型务 ...

  2. Java开发中常用jar包整理及使用

    本文整理了我自己在Java开发中常用的jar包以及常用的API记录. <!-- https://mvnrepository.com/artifact/org.apache.commons/com ...

  3. paip.java 开发中web server的选择jboss resin tomcat比较..

    paip.java 开发中web server的选择jboss resin tomcat比较.. 作者Attilax  艾龙, EMAIL:1466519819@qq.com 来源:attilax的专 ...

  4. Java开发中常见的危险信号(中)

    本文来源于我在InfoQ中文站原创的文章,原文地址是:http://www.infoq.com/cn/news/2013/12/common-red-flags-in-java-1 Dustin Ma ...

  5. Java开发中文件读取方式总结

    JAVA开发中,免不了要读文件操作,读取文件,首先就需要获取文件的路径. 路径分为绝对路径和相对路径. 在文件系统中,绝对路径都是以盘符开始的,例如C:\abc\1.txt. 什么是相对路径呢?相对路 ...

  6. java开发中遇到的问题及解决方法(持续更新)

    摘自 http://blog.csdn.net/pony12/article/details/38456261 java开发中遇到的问题及解决方法(持续更新) 工作中,以C/C++开发为主,难免与其他 ...

  7. Java开发中常见的危险信号(上)

    本文来源于我在InfoQ中文站原创的文章,原文地址是:http://www.infoq.com/cn/news/2013/12/common-red-flags-in-java-1 Dustin Ma ...

  8. 完整java开发中JDBC连接数据库代码和步骤[申明:来源于网络]

    完整java开发中JDBC连接数据库代码和步骤[申明:来源于网络] 地址:http://blog.csdn.net/qq_35101189/article/details/53729720?ref=m ...

  9. Java 开发中的对象拷贝

    前言 在 Java 开发中,很多时候需要将两个属性基本相同的对象进行属性复制,比如 DO 转 VO等等. 本文主要介绍自己实现的简易拷贝工具类与 Spring 提供的属性拷贝的对比. Spring 提 ...

随机推荐

  1. netty心跳机制测试

    netty中有比较完善的心跳机制,(在基础server版本基础上[netty基础--基本收发])添加少量代码即可实现对心跳的监测和处理. 1 server端channel中加入心跳处理机制 // Id ...

  2. 目前微信 微博 新浪 豆瓣等所有分享的js插件

    原理 功能 集成微信.微博.开心.豆瓣.人人.qq微博.搜狐.qq空间等分享 即时分享: 默认加载插件,即启动全部分享 定制分享:通过参数配置.静态数据配置 由你决定何时分享,如何分享 扩展: 通过数 ...

  3. 【JAVAWEB学习笔记】29_文件的上传------commons-fileupload

    今天内容: 文件的上传------commons-fileupload 文件上传和下载的实质:文件的拷贝 文件上传:从本地拷贝到服务器磁盘上   客户端需要编写文件上传表单---->服务端需要编 ...

  4. 使用SQL语句使数据从坚向排列转化成横向排列(排班表)

    知识重点: 1.extract(day from schedule01::timestamp)=13 Extract 属于 SQL 的 DML(即数据库管理语言)函数,同样,InterBase 也支持 ...

  5. POI使用:解析xls/xlsx文件(兼容office2003/2007/2010版本)

    package cn.eguid; import java.io.Closeable; import java.io.File; import java.io.FileInputStream; imp ...

  6. sqlmap详细使用 [精简]

    1. 基础用法: 一下./sqlmap.py 在kali和backtrack中使用sqlmap的时候,直接用:sqlmap ./sqlmap.py -u “注入地址” -v 1 –dbs   // 列 ...

  7. C# 禁止ALT+F4(钩子)

    1. Windows Forms中禁用窗体的关闭按钮  添加必要的命名空间: using System.Runtime.InteropServices;   添加必要的常数和API函数的引用 priv ...

  8. [转] DDD领域驱动设计框架分享

    从去年10月份开始,学了几个月的领域驱动设计(Domain Driven Design,简称DDD).主要是学习领域驱动设计之父Eric Evans的名著:<Domain-driven desi ...

  9. Rails核心组件

    Action Pack 包含Action Controller,Action View,Action Dispatch 封装了MVC的VC功能 Action Mailer 开发电子邮件服务的框架 Ac ...

  10. 关于java反射获取泛型

    public class Test<T> { private final TypeToken<T> typeToken = new TypeToken<T>(get ...