Java中list集合的clean()方法滥用引发的bug

Wesley13
• 阅读 688

做的是电商系统,主系统生成订单后,分别加入到不同的队列中给另外的三个子系统来异步处理,订单和商品是一对多的关系,在实际测试中,发现其中有一个子系统从队列中获取到的订单实体中,商品列表一直为空,子系统的开发人员一直说是获取不到值导致。因为这个问题,测试一直没通过,不得已只能打日志,反复排查。最后发现在这个子系统中,从队列拿到订单实体后,商品列表是不为空的,但是在执行了一个方法后,该订单实体类的商品就为空了。定位到问题后,调试这个方法,后来发现,整个方法的逻辑都是没问题的,只是在方法的最后,对传入的参数列表执行了clean()的操作。

     因为实际项目的业务场景比较复杂,在这里就设计一个简单的例子来实际说明。定义一个City 的实体类和Country的实体类

public class City {

private Integer id;

private String name;

    //省略了set和get方法

}

public class Country {

private Integer id;

private String name;

private List listCity; //Country和City是一对多的关系

    //省略了set和get方法

}

然后写测试类运行:

public class Test {

public static void main(String[] args) {

City city1=new City();

city1.setId(1);

city1.setName("广州");

City city2=new City();

city2.setId(2);

city2.setName("上海");

List listCity=new ArrayList<>();

listCity.add(city1);

listCity.add(city2);

Country country=new Country();

country.setId(1);

country.setName("中国");

country.setListCity(listCity);

getCity(country.getListCity());

System.out.println("list size:"+country.getListCity().size());

}

public static void getCity(List listCity){

listCity.clear();

}

}

运行该程序,运行结果是:list size:0

而如果把getCity方法中listCity.clear()注释掉,则运行结果为:list size:2

这正是导致项目出现那个bug的原因,其实一般来说,除非已经确定没有程序用到该集合,否则一般不会在方法的最后执行clear()操作。

知道这个原因后,现在来分析为什么调用这个方法后会导致这个后果。查看JDKclean()源码:

 Java中list集合的clean()方法滥用引发的bug

通过读源码发现,当调用clean()后,会将集合中的数据全部移走,并且将集合中长度置为0。也就是说该方法调用后,集合会变为空。

 Java中list集合的clean()方法滥用引发的bug

微信扫一扫
关注java高高手

点赞
收藏
评论区
推荐文章
blmius blmius
3年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
皕杰报表之UUID
​在我们用皕杰报表工具设计填报报表时,如何在新增行里自动增加id呢?能新增整数排序id吗?目前可以在新增行里自动增加id,但只能用uuid函数增加UUID编码,不能新增整数排序id。uuid函数说明:获取一个UUID,可以在填报表中用来创建数据ID语法:uuid()或uuid(sep)参数说明:sep布尔值,生成的uuid中是否包含分隔符'',缺省为
待兔 待兔
4个月前
手写Java HashMap源码
HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程22
Jacquelyn38 Jacquelyn38
3年前
2020年前端实用代码段,为你的工作保驾护航
有空的时候,自己总结了几个代码段,在开发中也经常使用,谢谢。1、使用解构获取json数据let jsonData  id: 1,status: "OK",data: 'a', 'b';let  id, status, data: number   jsonData;console.log(id, status, number )
Easter79 Easter79
3年前
thinkcmf+jsapi 实现微信支付
首先从小程序端接收订单号、金额等参数,然后后台进行统一下单,把微信支付的订单号返回,在把订单号发送给前台,前台拉起支付,返回参数后更改支付状态。。。回调publicfunctionnotify(){$wechatDb::name('wechat')where('status',1)find();
Wesley13 Wesley13
3年前
Java日期时间API系列31
  时间戳是指格林威治时间1970年01月01日00时00分00秒起至现在的总毫秒数,是所有时间的基础,其他时间可以通过时间戳转换得到。Java中本来已经有相关获取时间戳的方法,Java8后增加新的类Instant等专用于处理时间戳问题。 1获取时间戳的方法和性能对比1.1获取时间戳方法Java8以前
Easter79 Easter79
3年前
Twitter的分布式自增ID算法snowflake (Java版)
概述分布式系统中,有一些需要使用全局唯一ID的场景,这种时候为了防止ID冲突可以使用36位的UUID,但是UUID有一些缺点,首先他相对比较长,另外UUID一般是无序的。有些时候我们希望能使用一种简单一些的ID,并且希望ID能够按照时间有序生成。而twitter的snowflake解决了这种需求,最初Twitter把存储系统从MySQL迁移
Stella981 Stella981
3年前
Django中Admin中的一些参数配置
设置在列表中显示的字段,id为django模型默认的主键list_display('id','name','sex','profession','email','qq','phone','status','create_time')设置在列表可编辑字段list_editable
Wesley13 Wesley13
3年前
JPA(三)之实体关系一对多(多对一)
1.背景介绍:  对于购买商品时,订单信息(Order)和订单商品信息(OrderItem)的关系就是一对多的关系。2.实体bean:  Order.java代码packagecom.sinoi.bean;importjava.util.HashSet;importjava.util.Set;
Python进阶者 Python进阶者
10个月前
Excel中这日期老是出来00:00:00,怎么用Pandas把这个去除
大家好,我是皮皮。一、前言前几天在Python白银交流群【上海新年人】问了一个Pandas数据筛选的问题。问题如下:这日期老是出来00:00:00,怎么把这个去除。二、实现过程后来【论草莓如何成为冻干莓】给了一个思路和代码如下:pd.toexcel之前把这