Java堆栈区对内存的影响

## 例子

String[] arr0 = {"1", "2", "3", "4"};
String[] arr1 = {"5", "6", "7", "8"};
String[] arr2 = {"9", "10", "11", "12"};
String[] arr3 = {"13", "14", "15", "16"};

List<String[]> list = new ArrayList<>();
String[] arr = new String[4];
for(int i = 0; i < arr1.length; i++){
    arr[0] = arr0[i];
    arr[1] = arr1[i];
    arr[2] = arr2[i];
    arr[3] = arr3[i];
    list.add(arr);
}

这段代码的执行结束后,list的结果是什么呢?

答案:

[4,8,12,16]
[4,8,12,16]
[4,8,12,16]
[4,8,12,16]

但是,我们期望的list执行结果是将每个对应的元素都加进去。为什么没有按照我们期望执行呢?

## 原因

Java在创建对象时,将对象的引用放在栈里,而将对象放在堆中。list加进去的arr是一个引用,而arr指向堆中的对象new Array[4]。每次加进去的都是同一个引用指向的同一个对象,因此,当该唯一对象的值改变时,其list内所用内容的值也跟着改变。

第一次循环执行结束的结果

第二次循环执行结束的结果

第三次循环执行结束的结果

第四次循环执行结束的结果

## 改进

如何获得我们想要的结果呢。先上代码:

String[] arr0 = {"1", "2", "3", "4"};
String[] arr1 = {"5", "6", "7", "8"};
String[] arr2 = {"9", "10", "11", "12"};
String[] arr3 = {"13", "14", "15", "16"};

List<String[]> list = new ArrayList<>();
String[] arr = null;
for(int i = 0; i < arr1.length; i++){
    arr = new String[4];
    arr[0] = arr0[i];
    arr[1] = arr1[i];
    arr[2] = arr2[i];
    arr[3] = arr3[i];
    list.add(arr);
}

可以在每次循环时创建新的对象。新的对象被赋值时,指向新的常量池数据。这样list中每次加进来的引用虽然时同一个,到那时指向的对象每次都不一样。这样就达到了改变list中内容的效果。

上图解释:

经验分享 程序员 微信小程序 职场和发展