初学java,请问为什么将ArrayList对象get方法的返回值赋给一个引用,操作这个引用不会影响ArrayList对象内对应元素呢?
List list = new ArrayList();
list.add("abc");
String stri = list.get(0);
stri += "def";
System.out.println(stri);
System.out.println(list);
输出结果为:
abcdef
[abc]
但是我看到的ArrayList的get方法就是在判断index是否超范围之后直接return (E) elementData[index];为什么对引用stri的操作不会改变list的元素呢?
刚开始学java,很多东西没弄清楚,求前辈指点一下.m(__)m
主要是String类型的数据其实是不可变的,
stri += "def";
分解成
stri = stri + "def"即 stri = "abc" + "def"
这步操作,你要这样理解:
假定开始的时候内存中一块空间A存放“abc”,并且名字是stri
可以表示为: stri -> A("abc")
而对于String类型来说,他已经分配的内存是不能改动的,也就是不会在A空间的“abc”后面追加内容,所以做加法操作的时候,会重新分配一块内存B来存放计算结果,
也就是B("abcdef")
并将内存B地址赋值给stri,
显然,stri所引用的内存地址都变了,stri已经不是原先list里面的那个了。
反而是subList方法,return后面是new Sublist(this, fromIndex, toIndex);但
List subList = list.subList(0, 1);
subList.set(0, "def");
System.out.println(subList);
System.out.println(list);
输出结果为:
[def]
[def]
我觉得subList指向一个新对象啊为什么会影响到原来的list呢?
画个内存图就明白了
还有String对象是不可变的
其实你就想着 list.get(0)是取list的第一个值 list.get(1) 是取第二个值得 简单说说list.get() 也可以理解为取值 不影响 lis的值
因为 stri += "def"; 改变了stri的引用了
String 是 final的。所以值是不变的。值改变就会产生新对象。所以你输出的并不是原来的stri了,而是系统新new的对象。
这都是属于两个对象了,里面对应的内存地址都不一样,都是属于自己的各自的数据,互相不会有影响的。