一维数组的内存分析
首先要了解内存的结构:
- 栈:存放局部变量
- 堆:存放new实例化的对象、数组等
- 方法区:包括常量池、静态区
声明初始化数组
int arr = new int[1,2,3];
String[] arr1 = new String[4];
arr1[1] = "刘德华";
arr1[2] = "张学友";
一维数组的内存分配如下图
变量arr
和arr1
存放在栈区,而通过new
开辟的数组存储空间则在堆区,arr
数组是int
类型数组,所有堆区开辟3个空间并且初始化为0,紧接着会给赋值为指定的1,2,3。然后栈区的arr
的首地址会指向0x34ab
,这样就可以通过arr
来访问堆区开辟的内存空间,进而访问到里面的元素。
声明arr1
数组时也一样,在栈区存放变量arr1
,在堆区开辟相应的内存空间,字符串类型数组默认初始化的值为null
,然后在进行赋值操作。arr1
首地址会指向0x12ab
,但是赋值的字符串"刘德华"、“张学友”是存放在字符串常量区的。这样就可以通过arr1
来访问堆区开辟的内存空间,进而访问到里面的元素。
当arr1
重新指向新开辟的内存空间时,原有的在堆区开辟的内存空间,会根据引用计数算法,在合适的时候由垃圾回收机制自动回收内存。
二维数组的内存分析
二维数组的初始化默认值如下:
int[][] arr2 = new int[4][3];
System.out.println(arr2);
System.out.println(arr2[0]);
System.out.println(arr2[0][0]);
输出结果:
[[I
表示二维数组并且是int类型,@
后面的值指的是地址值。
[I
表示一维数组并且是int类型,@
后面的值指的是地址值。
0
是因为arr2[0][0]
指的就是int
类型元素,所以初始化默认值为0
。
注意下面这种情况:
int[][] arr3 = new int[4][];
System.out.println(arr3);
System.out.println(arr3[0]);
System.out.println(arr3[0][0]);
输出结果:
输出arr3
时,相当于输出二维数组的首地址,而初始化arr3 = new int[4][]
表示arr3
指向了一个二维数组,其实也就是指向了一个一维数组,只不过这个一维数组的元素类型又是一维数组。
所以输出arr3[0]
时输出为null
,但是输出arr3[0][0]
时会报空指针异常,因为无法访问空指针内存空间。
二维数组内存分配情况
int[][] arr1 = new int[4][3];
arr1[1] = new int[]{1,2,3};
arr1[2] = new int[4];
arr1[2][1] = 30;
首先在栈区分配空间给arr1
,然后在堆区开辟地址为0x1234
的一个一维数组,但这个一维数组的元素还是一个一维数组,所以初始化默认为null
。
然后给arr1[1]
赋值,在堆区开辟一个地址为0x7788
的一个一维int
类型数组,所以初始化默认为0
,紧接着会赋值为1,2,3。
然后给arr1[2]
在堆区开辟一个大小为4
的动态数组,地址为0x6677
,初始化默认值为0
。
最后arr1[2][1]
赋值为30
。