前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >List.add 方法添加元素时只会添加最后一条元素的问题与解决

List.add 方法添加元素时只会添加最后一条元素的问题与解决

作者头像
时间静止不是简史
发布2022-04-02 08:16:29
1.7K0
发布2022-04-02 08:16:29
举报
文章被收录于专栏:Java探索之路Java探索之路

List.add 方法添加元素时只会添加最后一条元素的分析解决


前言

在之前编写业务代码时, 遇到了一个比较神奇的现象, 如标题中描述的那样: 在对list 集合使用 add/set 方法并且遍历的去添加对象时, 只会添加最后一个元素的问题 . 下面就进行简单的分析,

一、问题描述

现有一个需求: 在请求时携带一个map, 该map里面的key有几个. 那么, 在返回结果集 map中就需要将对应的 key 和 value 捞出来一起展示, 如果结果不止一个map, 则将这些map放入一个list中

模拟代码片如下

结果演示 可以看到, 在上面结果集 resultListMap遍历过程中, 我们根据请求 requestMap的key 去结果集中寻找key对应的value 然后放入 responseMap, 因为结果集map 不止一个, 所以将其放入list 中. 但是,

在这里插入图片描述
在这里插入图片描述

二、原因分析

1.简化分析

  1. 为了将原来的问题简化, 我们可以在把问题简化下, 下面是简化之后的代码
代码语言:javascript
复制
        List<User> userList = new ArrayList<>(2);
        User u1 = new User();
        for (int i = 0; i < 5; i++) {
            u1.setId(i);
            u1.setPwd("odd taxi"+ i);
            u1.setUsername("小早川"+ i);
            userList.add(u1);
        }

        for (User user : userList) {
            System.out.println("user = " + user);
        }
  1. 由下图可知 list.add 通过遍历去调用, 又是只加入最后一条元素, 而且遍历几遍就有几条相同的元素
在这里插入图片描述
在这里插入图片描述
  1. 对上述代码进行断点调试 由我们对每次循环对象属性和 userList 中属性对比可知: 在每次循环结束后, userList 中所有的数据都会变成最后一次遍历数据*遍历次数
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

结合上面问题和调试结果, 以及资料可知: 1. List 中的 add, set 方法在添加对象(Object) 或者是集合(Collection)时, 添加的是对对象的引用 因此, 如果在循环外声明要保存的对象或集合, 但是却在循环内赋值的话, 就会导致上述问题 2. 因为在循环外对象或者集合只声明了一次, 因此无论如何赋值, 只会保存最后一次赋值. 而在循环内 list.add 方法添加的实际上只相当于对最后一次插入的对象或者集合的引用

  1. 基于上述分析, 我们先修改简化版代码: 只需将对象初始化放在循环内即可
代码语言:javascript
复制
        List<User> userList = new ArrayList<>(2);
        User u1 ;
        for (int i = 0; i < 5; i++) {
            u1 = new User();
            u1.setId(i);
            u1.setPwd("odd taxi"+ i);
            u1.setUsername("小早川"+ i);
            userList.add(u1);
        }

        for (User user : userList) {
            System.out.println("user = " + user);
        }

2.回归本题

根据上述思路我们去解决实际问题, 对比下修改前和修改后的代码

  1. 修改前
代码语言:javascript
复制
		//---------------------修改前---------------------
        //模拟请求
        Map<String, Object> requestMap = new HashMap<>();
        requestMap.put("id", "");
        requestMap.put("passwd", "");
        requestMap.put("userName", "");
        //模拟结果集
        Map<String, Object> resultMap = new HashMap<>();
        resultMap.put("id", "1");
        resultMap.put("passwd", "odd taxi");
        resultMap.put("userName", "小早川");
        Map<String, Object> resultMap2 = new HashMap<>();
        resultMap2.put("id", "2");
        resultMap2.put("passwd", "Brazilian war dance");
        resultMap2.put("userName", "白川");
        List<Map<String, Object>> resultListMap = new ArrayList<>();
        resultListMap.add(resultMap);
        resultListMap.add(resultMap2);
        //模拟响应
        List<Map<String, Object>> responseListMap = new ArrayList<>();
        Map<String, Object> responseMap = new HashMap<>();
        // 实现: 根据请求map中的 key, 返回所有结果集中含有请求map所带k的list.map集合
        for (Map<String, Object> resMap : resultListMap) {
            requestMap.forEach((k, v) -> {
                responseMap.put(k, resMap.get(k));
            });
            responseListMap.add(responseMap);
        }
        System.out.println("responseListMap = " + responseListMap);
  1. 修改后
代码语言:javascript
复制
        //---------------------修改后---------------------
        //模拟请求
        Map<String, Object> requestMap = new HashMap<>();
        requestMap.put("id", "");
        requestMap.put("passwd", "");
        requestMap.put("userName", "");
        //模拟结果集
        Map<String, Object> resultMap = new HashMap<>();
        resultMap.put("id", "1");
        resultMap.put("passwd", "odd taxi");
        resultMap.put("userName", "小早川");
        Map<String, Object> resultMap2 = new HashMap<>();
        resultMap2.put("id", "2");
        resultMap2.put("passwd", "Brazilian war dance");
        resultMap2.put("userName", "白川");
        List<Map<String, Object>> resultListMap = new ArrayList<>();
        resultListMap.add(resultMap);
        resultListMap.add(resultMap2);
        //模拟响应
        List<Map<String, Object>> responseListMap = new ArrayList<>();
        // 实现: 根据请求map中的 key, 返回所有结果集中含有请求map所带k的list.map集合
        for (Map<String, Object> resMap : resultListMap) {
            // 问题修改: 将 list 需要 add 的对象/集合 放到循环内进行初始化!!!
            Map<String, Object> responseMap = new HashMap<>();
            requestMap.forEach((k, v) -> {
                responseMap.put(k, resMap.get(k));
            });
            responseListMap.add(responseMap);
        }
        System.out.println("responseListMap = " + responseListMap);
  1. 可以看出仅仅是将list 中需要 add 的集合 responseMap 的初始化放到循环内, 问题就得到解决, 如下图所示
在这里插入图片描述
在这里插入图片描述

总结

1. List 中的 add, set 方法在添加对象(Object) 或者是集合(Collection)时, 添加的是对对象的引用 2. 在循环外声明对象或集合, 在循环内使用list.add 就会导致list 中引用的数据地址全部都是最后一次添加的元素地址 如果想要避免. 只需要将被add的对象/集合在循环内初始化即可


参考: https://blog.csdn.net/ww77889126/article/details/84952631

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • List.add 方法添加元素时只会添加最后一条元素的分析解决
  • 前言
  • 一、问题描述
  • 二、原因分析
    • 1.简化分析
      • 2.回归本题
      • 总结
      领券
      问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档