错误:
用list存储User对象信息,当存储多个对象时,发现存储的数据都是一样的
之前代码:
User user = new User();
List<User> list = new ArrayList<>();
for (int i = 0; i < 5; i++) {
user.setName("Tom"+i);
user.setAge("18"+i);
user.setLocation("China"+i);
list.add(user);
}
try {
userDao.insert(list);
} catch (Exception e) {
e.printStackTrace();
}
通过调用insert方法,发现添加的数据都是一样的,添加的都是同一个数据
原理:
简单来讲:
在JVM栈中创建了一个User类型的地址(地址通常为十六进制的数字,这里假设为0x12345678)
然后使用new关键字在JVM堆中开辟了一个新空间,将User对象存储到这个堆中
当i=0时,执行user.setName(“Tom”+0)…通过地址值找到堆中的User对象,将对象中的属性分别赋值,然后执行list.add方法,就是将User的地址存储到List集合中
通过ArrayList源码可知,它的底层数据结构就是Object类型的数组,所以这里的add方法就是将地址存到一个数组中,我们访问集合中的数据,其实就是读取这个数组中地址所指向的值
当i=1时,执行user.setName(“Tom”+1),此时因为并没有使用new关键字去创建新的对象,所以地址指还是指向原来的。将原来的属性值进行重新赋值。然后执行list.add方法,又将User的地址存储到List集合中,此时ArrayList的数组中就存储了两个相同的User对象,并且我们去访问,得到的属性值都是一样的。
当i=2,i=3,i=4以此类推,Object数组中就有五个相同地址的对象,而且属性值均为最新的赋值
了解清楚底层原理后,问题很快得到解决
我们只需要在外部声明对象,每进行一次循环的时候,都new一个新的对象,这样就可以保证每次添加集合的地址均不相同,也不会被覆盖
User user;
List<User> list = new ArrayList<>();
for (int i = 0; i < 5; i++) {
user = new User();
user.setName("Tom"+i);
user.setAge("18"+i);
user.setLocation("China"+i);
list.add(user);
}
try {
userDao.insert(list);
} catch (Exception e) {
e.printStackTrace();
}
总结:
一句话概括就是list.add()添加数据插入的是对象所指向的地址,如果插入同一个对象就会出现覆盖掉之前的数据,出现重复的两份数据。