数组的概述
数组的特点:
1、数组是有序排列的;
2、数组属于引用数据类型的变量。数组的元素,既可以是基本数据类型,也可以是引用数据类型;
3、创建数组对象会在内存中开辟一整块连续的空间;
4、在数组初始化时,其长度便确定了;长度一旦确定,就不能修改了。
数组的声明和初始化
一维数组
一维数组的声明和初始化:
int[] ids;//声明
int[] ids2 = {1,2,3,4};
int ids3[];
//静态初始化:数组的初始化和数组元素的赋值操作同时进行
ids=new int[]{1001,1002,1003,1004};
//动态初始化:数组的初始化和数组元素的赋值操作分开进行
String[] names = new String[5];
names[0] = "100";
一维数组的初始值
由于数组属于引用数据类型变量,所以数组元素的初始值和其引用的数据类型初始值一致;数据类型初始值如下表:
数据类型 | 初始值 |
---|---|
byte、short、int、long | 0 |
float、double | 0.0 |
boolean | false |
char | ASCII码为0对应的字符 |
String | null |
注:char数据类型的初始值为ASCII码等于0的字段即空格,此空格不等于null,具体如下图所示:
一维数组内存结构解析
1、先介绍一下内存简化结构(JVM),下图中仅介绍主要结构,便于更好的理解数组,详细结构的待学(●’◡’●),如下为个人学习后的理解,仅供参考,如有不对,请指正!谢谢。
2、当我们声明一个一维数组变量时,会在栈中创建一个区域,存储局部变量,而在初始化时,会在堆中申请相对应的数组空间,对其赋初值,然后将堆中的地址放至栈相对应的局部变量的位置,使两者产生关联,具体如下图:
3、当对数组中的元素赋值时,其实是将值存储至常量池,然后将值所在的地址传至队中元素的位置,此时便完成了数组的声明、初始化、赋值。
二维数组
二维数组的声明和初始化
//静态初始化
int[][] arr1 = new int[][] {{1,2,3},{4,5},{6,7,8,9,10}};
//动态初始化1
String[][] arr2 = new String[3][4];
//动态初始化2
String[][] arr3 = new String[4][];
//初始化其他写法
int[] arr4[] = new int[][] {{1,2,3},{4,5},{6,7,8,9,10}};
int[] arr5[] = {{1,2,3},{4,5},{6,7,8,9,10}};
二维数组的初始值
其实,二维数组外层元素的初始值和一维数组的初始值的情况一致,但其内层元素的初始值与其初始化方式相关。
1、当二维数组声明及初始化方式为:int[][] arr = new int[3][4]时,
内层(即行)元素的初始值为地址值
2、当二维数组声明及初始化方式为:int[][] arr = new int[3][]时,
内层(即行)元素的初始值为null
具体解释参考二维数组内存结构解析
二维数组内存结构解析
1、当我们声明一个二维数组变量时,会在栈中创建一个区域,存储该局部变量,而在初始化时,会在堆中申请相对应的数组空间,对其赋初值,然后将内层元素在堆中的地址放至栈相对应的局部变量的位置,使两者产生关联,然后再根据外层元素个数在堆中重新申请一整块连续的空间,并将地址放置堆中内层元素的位置中,具体如下图:
上图中为当二维数组声明及初始化方式为:int[][] arr = new int[3][4]时,
内层(即行)元素的初始值为地址值
2、当不知外层元素的个数为多少时,内存无法开出一块连续的空间,所以也不存在外层元素地址值,因此此时内层元素值为null,如下图:
执行代码测试:
数组的赋值和复制
赋值和复制的区别是在内存空间中是否重新开辟了一块新的空间,先看代码:
String[] str = new String[] {"A","B","C"};
String[] str1;
//赋值
str1 = str;
String[] str2 =new String[str.length];
//复制
for(int i=0;i<str.length;i++) {
str2[i] = str[i];
解读:赋值只是在栈中加入一个局部变量str1,然后将str中存储的地址值给str1,即str和str1指向的是堆中同一块区域;而复制是指不仅在栈中加入局部变量str2,而且在堆中开辟一块连续的空间,其长度和str一致,然后将数组str中元素值复制给str2中的元素。