三、Java 数组
数组是编程语言中常见的一种数据结构,用于存储多个数据,每个数组元素存放一个元素,通常可通过数组元素的索引来访问数组元素,访问包括维数组元素赋值和取出数组元素的值。
3.1理解数组——数组也是一种类型
- Java 的数组要求所有数组元素具有相同的数据类型,即一个数组中只能存储一种数据类型的数据。
- 一旦数组的初始化完成,数组在内存中所占的空间将被固定下来。
- Java 数组既可以存储基本类型的数据,也可以存储引用类型的数据。
- 数组也是一种数据类型,它本身就是一种引用类型,例 int 是一个基本类型,但 int[] 就是一种引用类型。
3.2定义数组
Java 语言支持两种语法格式来定义数组:
type [] arrayName;
type arrayName[];
数组是一种引用类型的变量,在使用数组**定义一个变量时,仅仅表示定义了一个引用变量**(在这里可以理解为指针),这个引用变量还**未指向任何有效的内存**,**因此定义数组时不能指定数组长度。**由于没有指向任何有效的内存空间(**相当于这个数组还没有分配内存)**,因此这个数组还不能使用,**只有对数组初始化后才能使用。**
3.3数组的初始化
继续上文,数组初始化后才能使用,而初始化就是**为数组的元素分配内存,并为每个数组元素赋初始值。**
初始化有两种方式:
3.3.1静态初始化
初始化时由程序员显式指定每个数组元素的初始值,由系统决定数组长度。
public static void main(String[] args) {
int [] intArr;
intArr = new int[]{1,2,3,4};
int [] a = {1,2,3,4};
System.out.println(intArr);
System.out.println(Arrays.toString(intArr));
System.out.println(a[0]);
}
3.3.2动态初始化
初始化时程序员只指定数组长度,由系统给数组元素分配初始值。
- 系统有如下规则来进行元素分配初始值:
- 数组元素类型是基本类型中的整数类型(byte、short、int 和 long),则数组元素的值是 0;
- 数组元素类型是基本类型中的浮点类型(float、double),则数组元素的值是 0.0;
- 数组元素类型是基本类型中的字符类型 (char),则数组元素的值是‘\u0000’;
- 数组元素类型是基本类型中的布尔类型(boolean),则数组元素的值是 false ;
- 数组元素类型是引用类型(类、接口和数组),则数组元素的值是 null 。
public static void main(String[] args) {
int [] a = new int[5];
float [] b = new float[5];
char [] c = new char[5];
boolean [] d = new boolean[5];
System.out.println(a[0]);
System.out.println(b[0]);
System.out.println(c[0]);
System.out.println(d[0]);
}
res:
在 Idea 中输出char类型数组的初始值,显示为乱码,在这里猜测的原因可能是 Idea不支持 Unicode 编码的原因( ‘\u0000’ 是属于 Unicode 编码)。
3.4使用数组
数组使用最常用的是**访问数组元素**,在 Java 语言中数组索引是从 0 开始的,即,第一个数组元素索引值为 0,最后一个元素索引值为数组长度减 1。**如果访问数组元素索引值小于0,或者大于等于数组长度**,**编译程序不会出现任何错误,但运行时会出现异常。**代码如下:
public static void main(String[] args) {
int [] a = new int[]{1,2,3,4};
System.out.println(a[1]);
System.out.println(a[-1]);
}
3.5 foreach 循环
除了使用 for 循环和 Arrays 类通过转换为 String 类型输出数组,在这里还有一种更简单的循环,foreach 循环;使用 foreach 循环遍历数组和集合元素时,无须获得其长度,无须根据索引来访问元素。代码如下:
public static void main(String[] args) {
String [] a = new String[]{"BGR的","foreach","测试"};
for (String i :a)
{
System.out.println(i);
}
}
3.6深入数组——(内存中的数组)
数组在前面已经介绍过了是一种引用数据类型,数组引用变量只是一个引用,**数组元素和数组变量在内存中是分开存放的。**下面将介绍数组在内存中的运行机制。
-
数组引用变量只是一个引用,这个引用变量可指向任何有效的内存,只有当该引用指向有效内存后,才可通过该数组变量来访问数组元素。 -
实际的数组对象被存储在堆(heap)内存中;如果引用该数组对象的数组引用变量是一个局部变量那么它被存储在栈(stack)内存中。 -
而我们则通过 p[index] 的形式实现访问存储在堆内存中的数组元素。 -
如果堆内存中的数组不再有任何引用变量指向自己,则这个数组将成为垃圾,该数组所占内存将被 Java 的垃圾回收机制回收。
接下来给大家看一个数组长度可以改变的假象:
java
public static void main(String[] args) {
// 分别使用动、静态初始化并定义数组
int [] a ={5,7,20};
int [] b = new int[4];
//输出a、b数组长度为了和后面对比
System.out.println("a数组的长度为:"+a.length);
System.out.println("b数组的长度为:"+b.length);
//分别遍历a、b数组
for (int i: a) {
System.out.print(i+" ");
}
System.out.println();
for (int i: b) {
System.out.print(i+" ");
}
System.out.println();
//将a赋给b
b = a;
//此时发现b数组的长度变成a数组的长度了
System.out.println("b数组的长度为:"+b.length);
//最后看一下b数组各个元素的值
for (int i: b) {
System.out.print(i+" ");
}
}
/*res:
a数组的长度为:3
b数组的长度为:4
5 7 20
0 0 0 0
b数组的长度为:3
5 7 20
读者需要牢记这个规则:定义并初始化一个数组后,会在内存中分配两个空间,一个用于存放数组的引用变量,另一个存放数组本身
如图可以看出,当执行 b = a 后,堆内存中第一个数组具有两个引用;a 变量和 b 变量都引用了第一个数组。**此时第二个数组失去引用变成垃圾,**等待垃圾回收机制进行回收。
|