Java Review (五、数组)

@


数组是编程语言中最常见的一种数据结构,可用于存储多个数据,每个数组元素存放一个数据,通 常可通过数组元素的索引来访问数组元素,包括为数组元素赋值和取出数组元素的值。

数组数据类型

Java的数组要求所有的数组元素具有相同的数据类型。因此,在一个数组中,数组元素的类型是唯一的,即一个数组里只能存储一种数据类型的数据,而不能存储多种数据类型的数据。
一旦数组的初始化完成,数组在内存中所占的空间将被固定下来,因此数组的长度将不可改变。即 使把某个数组元素的数据清空,但它所占的空间依然被保留,依然属于该数组,数组的长度依然不变。
Java的数组既可以存储基本类型的数据,也可以存储引用类型的数据,只要所有的数组元素具有相同的类型即可。
数组也是一种数据类型,它本身是一种引用类型。例如int是一个基本类型,但int[] 就是一种引用类型了。


声明数组变量

dataType[] arrayRefVar;   // 首选的方法
 
或
 
dataType arrayRefVar[];  // 效果相同,但不是首选方法

数组是一种引用类型的变量,因此使用它定义一个变量时,仅仅表示定义了一个引用变量(也就是 定义了一个指针),这个引用变量还未指向任何有效的内存,因此定义数组时不能指定数组的长度。而 且由于定义数组只是定义了一个引用变量,并未指向任何有效的内存空间,所以还没有内存空间来存储 数组元素,因此这个数组也不能使用,只有对数组进行初始化后才可以使用。

初始化数组

Java语言中数组必须先初始化,然后才可以使用。所谓初始化,就是为数组的数组元素分配内存空 间,并为每个数组元素赋初始值。
数组的初始化有如下两种方式。

  • 静态初始化:初始化时由程序员显式指定每个数组元素的初始值,由系统决定数组长度。
  • 动态初始化:初始化时程序员只指定数组长度,由系统为数组元素分配初始值。

静态初始化

arrayName = new type[]{elementl, element2 , element3 , element4 ...}

在上面的语法格式中,前面的type就是数组元素的数据类型,此处的type必须与定义数组变量时 所使用的type相同,也可以是定义数组时所指定的type的子类,并使用花括号把所有的数组元素括起 来,多个数组元素之间以英文逗号(,)隔开,定义初始化值的花括号紧跟[]之后。值得指出的是,执行 静态初始化时,显式指定的数组元素值的类型必须与new关键字后的type类型相同,或者是其子类的 实例。下面代码定义了使用这三种形式来进行静态初始化。

实例
//定义一个int数组类型的变量,变量名为intArr
int[] intArr;
//使用静态初始化,初始化数组时只指定数组元素的初始值,不指定数组长度
intArr = new int[](5, 6, 8, 20);
//定义一个Object数组类型的变量,变量名为:bjArr
Object[] objArr;
//使用静态初始化,初始化数组时数组元素的类型是
//定义数组时所指定的数组元素类型的子类
objArr = new String [ ] ( "Java", "C#" );
Object[] objArr2;
//使用静态初始化
objArr2 = new Object [ ] {"Java” ,"Python"};

动态初始化

动态初始化只指定数组的长度,由系统为每个数组元素指定初始值。动态初始化的语法格式如下:

arrayName = new type[length];

在上面语法中,需要指定一个int类型的length参数,这个参数指定了数组的长度,也就是可以容 纳数组元素的个数。与静态初始化相似的是,此处的type必须与定义数组时使用的type类型相同,或 者是定义数组时使用的type类型的子类。下面代码示范了如何进行动态初始化(程序清单同上)。

//数组的定义和初始化同时完成,使用动态初始化语法
int[] prices = new int[5];
//数组的定义和初始化同时完成,初始化数组时元素的类型是定义数组时元素类型的子类 Object[] books = new String[4];

执行动态初始化时,程序员只需指定数组的长度,即为每个数组元素指定所需的内存空间,系统将 负责为这些数组元素分配初始值。指定初始值时,

数组元素类型 系统分配初始值规则
整数类型(byte、short、int和long) 0
浮点类型(float、double) 0.0
字符类型(char) '\u0000'
布尔类型(boolean) false
引用类型(类、接口和数组) null

数组使用

Java语言的数组索引是从0开始的,也就是说,第一个数组元素的索引值为0,最后一个数组元素 的索引值为数组长度减1。

double[] arr = {1.9, 2.9, 3.4, 3.5};

arr[2]表示3.4,数组第三个元素。

for-each循环

Java 5之后,Java提供了一种更简单的循环:foreach循环,这种循环遍历数组和集合更加简洁。

深入数组

数组是一种引用数据类型,数组引用变量只是一个引用,数组元素和数组变量在内存里是分开存放的。

内存中的数组

数组引用变量只是一个引用,这个引用变量可以指向任何有效的内存,只有当该引用指向有效内存 后,才可通过该数组变量来访问数组元素。

与所有引用变量相同的是,引用变量是访问真实对象的根本方式。也就是说,如果希望在程序中访问数组对象本身,则只能通过这个数组的引用变量来访问它。

实际的数组对象被存储在堆(heap)内存中;如果引用该数组对象的数组引用变量是一个局部变量, 那么它被存储在栈(stack)内存中。

数组在内存中的存储示意图

在这里插入图片描述

如果堆内存中数组不再有任何引用变量指向自己,则这个数组将成为垃圾,该数组所占的内存将会 被系统的垃圾回收机制回收。因此,为了让垃圾回收机制回收一个数组所占的内存空间,可以将该数组 变量赋为null,也就切断了数组引用变量和实际数组之间的引用关系,实际的数组也就成了垃圾。

只要类型相互兼容,就可以让一个数组变量指向另一个实际的数组,这种操作会让人产生数组的长度可变的错觉。如下:

实例
public class ArrayInRam {

	public static void main(String[] args) {
		//静态初始化a
		int []a= {5,7,20};
		//动态初始化b
		int []b=new int[4];
		//输出b的长度
		System.out.println(b.length);
		//循环遍历输出a
		for(int aa:a) {
			System.out.print(aa+"\t");
		}	
		//b引用a
		b=a;
		//输出b的长度
		System.out.println("\n"+b.length);

	}

}

输出结果:

4
5	7	20	
3

看起来b数组的长度似乎发生了变化。实际上:

定义并初始化一个数组后,在内存 中分配了两个空间,一个用于存放 数组的引用变量,另一个用于存放数组本身。

  • 当程序定义并初始化了 a、b 两个数组后,系统内存中实际上产生了 4块内存区,其中栈内存中有 两个引用变量:a和b;堆内存中 也有两块内存区,分别用于存储a 和b引用所指向的数组本身。

可以非常清楚地看出a引用和b引用各自所引用的数组对象,并可以很清楚地看出a变量所引用的数组长度是3, b变量所引用的数组长度是4。

定义并初始化a、b两个数组后的内存示意图

在这里插入图片描述

  • 当执行代码 b = a 时,系统将会把a的值赋给b, a和b都是引用类型变量,存储的是地址。因此把a的值赋给 b后,就是让b指向a所指向的地址。

当执行7b = a;之后,堆内存中的第一个数组具有了两个引用:a变量和b变 量都引用了第一个数组。此时第二个数组失去了引用,变成垃圾,只有等待垃圾回收机制来回收它——但它的长度依然不会改变,直到它彻底消失。

让b引用指向a引用所指向数组后的存储示意图

在这里插入图片描述

基本类型数组的初始化

对于基本类型数组而言,数组元素的值直接存储在对应的数组元素中,因此,初始化数组时,先为 该数组分配内存空间,然后直接将数组元素的值存入对应数组元素中。

实例
public class PrimitiveArrayTest {
	public static void main(String[] args) {
	//定义一个int[]类型的数组变量iArr
	int[] iArr;
	//动态初始化数组,数组长度为5
	iArr = new int[5];
	//采用循环方式为每个数组元素赋值
	for (int i = 0; i <iArr.length ; i++ ){
	  iArr[i] = i + 10;
    }
 }
}

定义iArr数组变量后的存储示意图

在这里插入图片描述


动态初始化iArr数组变量后的存储示意图

在这里插入图片描述

显式指定每个数组元素值后的存储示意图

在这里插入图片描述

引用类型数组的初始化

引用类型数组的数组元素是引用,因此情况变得更加复杂。每个数组元素里存储的还是引用,它指 向另一块内存,这块内存里存储了有效数据。

先定义一个Person类:

public class Person {
	public int age; // 年龄
	public double height; // 身高
	//定义一个info方法
	public void info(){
	System.out .println ("我的年龄是:" + age+ "我的身高是:"+ height);
	}

}

定义一个Person]]数组,接着动态初始化这个Person[]数组,并为这个数组的每个数组元素指定值:

public class ReferenceArrayTest {

	public static void main(String[] args) {
		//定义一个students数组变量,其类型是Person[]
		Person[] students;
		//执行动态初始化
		students = new Person[2];
		//创建一个Person实例,并将这个Person实例赋给zhang变量
		Person zhang=new Person();
		//为zhang所引用的Person对象的age、height赋值
		zhang.age = 15;
		zhang.height = 158;
		//创建一个Person实例,并将这个Person实例赋给丄ee变量
		Person lee = new Person();
		//为lee所引用的Person对象的age、height赋值
		lee.age = 16;
		lee.height = 161;
		//将zhang变量的值赋给第一个数组元素
		students[0] = zhang;
		//将lee变量的值赋给第二个数组元素
		students[1] = lee;
		//下面两行代码的结果完全一样,因为lee
		//和students [1]指向的是同一个Person实例
		lee.info();
		students[1].info();
	}

}



定义students数组变量后的存储示意图

在这里插入图片描述


动态初始化students数组变量后的存储示意图

在这里插入图片描述


创建两个Person实例后的存储示意图

在这里插入图片描述


为数组元素赋值后的存储示意图

在这里插入图片描述

多维数组

Java语言里提供了支持多维数组的语法。

但从底层运行机制来讲,二维数组实际上仍然是一维数组。

Java语言里的数组类型是引用类型,因此数组变量其实是一个引用,这个引用指向真实的数组内存。 数组元素的类型也可以是引用,如果数组元素的引用再次指向真实的数组内存,这种情形看上去很像多维数组。
回到前面定义数组类型的语法:

type[] arrName;

这是典型的一维数组的定义语法,其中type是数 组元素的类型。如果希望数组元素也是一个引用,而且是指向int数组的引用,则可以把type具体成int[],那么上面定义数组的 语法就是 int[][] arrName

如果把int这个类型扩大到Java的所有类型(不包括数组类型),则出现了定义二维数组的语法:

type[][] arrName;

Java语言采用上面的语法格式来定义二维数组,但它的实质还是一维数组,只是其数组元素也是引 用,数组元素里保存的引用指向一维数组。

接着对这个"二维数组”执行初始化,同样可以把这个数组当成一维数组来初始化,把这个“二维 数组”当成一个一维数组,其元素的类型是type口类型,则可以釆用如下语法进行初始化:

arrName = new type[length][]

上面的初始化语法相当于初始化了一个一维数组,这个一维数组的长度是lengtho同样,因为这个 一维数组的数组元素是引用类型(数组类型)的,所以系统为每个数组元素都分配初始值:nullo
这个二维数组实际上完全可以当成一维数组使用:使用new type[length] 初始化一维数组后,相当 于定义了 length个type类型的变量;类似的,使用 new type[length][] 初始化这个数组后,相当于定义了 length个type[]类型的变量,当然,这些type[]类型的变量都是数组类型,因此必须再次初始化这些数组。

数组排序

几个常见数组排序方法(冒泡排序、插入排序、快速排序、希尔排序)







参考:
【1】:《疯狂Java讲义》
【2】:《Java核心技术 卷一》
【3】:https://www.runoob.com/java/java-array.html

posted @ 2020-06-05 21:53  三分恶  阅读(191)  评论(0编辑  收藏  举报