我们知道java中new方式创建的对象都是在堆中创建的,而局部变量对应的值存放在栈上。那么java中的int [] arr={1,2,3}是存放在什么地方的呢,int []arr = new int[3]又是存放在什么地方的呢,
下面我们通过编写两个小例子,通过查看生成的字节码文件,来了解jvm会如何来处理这两种情况的。
1.int[] arr = new int[3]示例
public classArrayTest {public static voidmain(String[] args) {int[] arr = new int[3];
}
}
生成的字节码文件如下:这里只显示主要信息。java对应的字节码指令信息请参考博文java 字节码指令
Compiled from "ArrayTest.java"
public classArrayTest {publicArrayTest();
Code:0: aload_0 //从局部变量0中加载引用到堆栈,即将当前对象引用压入栈。1: invokespecial #8 //调用对象的实例方法,就是初始化方法init, 这里#8表示的是对应在方法池里方法的引用,
4: return
public static voidmain(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 classArrayTest {public static voidmain(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."":()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++中数组是可以在栈中分配空间的。