透过字节码分析java基本类型数组的内存分配方式。
我们知道java中new方式创建的对象都是在堆中创建的,而局部变量对应的值存放在栈上。那么java中的int [] arr={1,2,3}是存放在什么地方的呢,int []arr = new int[3]又是存放在什么地方的呢,
下面我们通过编写两个小例子,通过查看生成的字节码文件,来了解jvm会如何来处理这两种情况的。
1.int[] arr = new int[3]示例
public class ArrayTest { public static void main(String[] args) { int[] arr = new int[3]; } }
生成的字节码文件如下:这里只显示主要信息。java对应的字节码指令信息请参考博文java 字节码指令
Compiled from "ArrayTest.java" public class ArrayTest { public ArrayTest(); Code: 0: aload_0 //从局部变量0中加载引用到堆栈,即将当前对象引用压入栈。 1: invokespecial #8 //调用对象的实例方法,就是初始化方法init, 这里#8表示的是对应在方法池里方法的引用, 4: return public static void main(java.lang.String[]); Code: 0: iconst_3 //将3压入栈。 1: newarray int //该指令首先从栈中弹出数据3,并创建一个大小为3的整形数组,并将该数组的对象引用(地址)压入栈中。 3: astore_1 //将弹出栈中对象引用存放到局部变量1中。 4: return }
2.int arr[] = {1,2,3}示例:
public class ArrayTest { public static void main(String[] args) { //int[] arr = new int[3]; int[] arr = {1,2,3}; } }
生成的字节码如下:
Compiled from "ArrayTest.java" public class ArrayTest { public ArrayTest(); Code: 0: aload_0 1: invokespecial #8 // Method java/lang/Object."<init>":()V 4: return public static void main(java.lang.String[]); Code: 0: iconst_3 1: newarray int //同上 3: dup //拷贝一份栈顶的值,并将其压栈, 4: iconst_0 5: iconst_1 6: iastore //将栈顶的1存放到数组的0索引的位置 7: dup 8: iconst_1 9: iconst_2 10: iastore //将栈顶2存放到数组的1索引的位置 11: dup 12: iconst_2 13: iconst_3 14: iastore //将栈顶3存放到数组2索引的位置。 15: astore_1 16: return }
通过生成的字节码我们可以看出两种数组的创建都是都过newarr关键字来创建的,jvm会在堆中为其开辟空间,并将数组的引用存放在栈中。
由此可知java中的数组是在堆中分配空间。但是在C,C++中数组是可以在栈中分配空间的。