一、数组的概念

1.1 概述

数组是一种容器,可以同时存放多个数据值。

1.2 数组的特点

  1. 数组是一种引用数据类型
  2. 数组当中的多个数据,类型必须统一
  3. 数组的长度在程序运行期间不可改变

二、数组的使用

2.1 初始化

2.1.1 动态初始化(指定长度)

  1. 数据类型[] 数组名称 = new 数据类型[数组长度];--------a
  2. //a行代码可拆分为两个步骤
  3. 数据类型[] 数组名称;
  4. 数组名称 = new 数据类型[数组长度];

2.1.2 静态初始化(指定内容)

  1. 数据类型[] 数组名称 = new 数据类型[] {元素1,元素2,元素3,...};--b
  2. //或为以下省略格式
  3. 数据类型[] 数据名称 = {元素1,元素2,元素3,...};----c
  4. //b 行代码可拆分为两个步骤
  5. 数据类型[] 数组名称;
  6. 数组名称 = new 数据类型[] {元素1,元素2,元素3,...};
  7. //c 行代码不可以拆分为两个步骤

2.1.3 默认值

使用动态初始化数组的时候,其中的元素会自动拥有一个默认值,规则如下:

数据类型 默认值
整数类型 0
浮点类型 0.0
字符类型 ‘\u0000’ (是不可见字符,不是空格)
布尔类型 false
引用类型 null

注意: 静态初始化其实也有默认值的过程,只不过系统自动马上将后面大括号中的值赋给了数组。

2.2 访问数组元素

  • 直接打印数组名称,得到的是数组对应的:内存地址哈希值。如:[I@75412c2f,其中[代表数组,I代表int类型,@75412c2f代表哈希值。

  • 访问数组元素的格式:

  1. 数组名称[索引值]
  • 索引值: 就是一个int数字,代表数组当中元素的编号。索引值从0开始,一直到数组的长度-1为止。

三、Java中的内存划分

Java的内存需要划分为5个部分:

3.1 栈(Stack)

存放的是方法中的局部变量,用于运行方法。

  1. 局部变量:方法的参数,或者方法{}内部的变量。
  2. 作用域:一旦超出作用域,立刻从栈内存中消失。

3.2 堆(Heap)

凡是new出来的东西,都在堆当中。

  1. 堆内存里的东西都有一个地址值:16进制。
  2. 堆内存里的数据都有默认值。

3.3 方法区(Method Area)

存储.class相关信息,包含方法的信息。(只是死信息,真正方法运行一定要在栈当中)

3.4 本地方法栈(Native Method Stack)

与操作系统相关。

3.5 寄存器(PC Register)

与CPU相关,性能非常高。

四、数组的内存图解

4.1 一个数组的内存图解

  1. 程序执行流程:
  2. 1. 编译.java文件,main方法进入方法区。
  3. 2. 执行.class文件,方法区中的main方法进入方法栈。
  4. 3. 创建数组,JVM会在堆内存中开辟空间,存储数组。
  5. 4. 数组在内存中会有自己的内存地址,以十六进制数表示。
  6. 5. 数组中有3个元素,默认值为0
  7. 6. JVM将数组的内存地址赋值给引用变量array,因为保存的是地址,而不是一个具体的数值,所以称为引用数据类型。
  8. 7. 当执行array[1] = 10,程序先通过array找到数组,再通过数组的地址,找到数组在堆内存中的位置,通过索引1得到数组中第二个数据在堆内存中的位置,将它的值改为10

数组 - 图1

4.2 两个数组的内存图解

数组 - 图2

4.3 两个引用指向同一个数组的内存图解

数组 - 图3

五、数组的常见操作

5.1 数组越界异常

访问数组中不存在的索引,程序运行后,将会抛出 ArrayIndexOutOfBoundsException 数组越界异常。

5.2 数组空指针异常

  1. public static void main(String[] args) {
  2. int[] arr = {1,2,3};
  3. arr = null;
  4. System.out.println(arr[0]);

数组必须进行new初始化才能使用其中的元素。如果只是赋值了一个null,没有进行new创建,那么将会发生:空指针异常 NullPointerException

5.3 获取数组的长度

  1. 数组名称.length

注意:数组一旦创建,程序运行期间,长度不可改变

数组 - 图4

5.4 数组的遍历

  • IDEA中:变量/常量/数组名称.fori —> 生成for循环语句。
  1. public class Demo04Array {
  2. public static void main(String[] args) {
  3. int[] array = { 15, 25, 30, 40, 50, 60, 75 };
  4. // 首先使用原始方式
  5. System.out.println(array[0]); // 15
  6. System.out.println(array[1]); // 25
  7. System.out.println(array[2]); // 30
  8. System.out.println(array[3]); // 40
  9. System.out.println(array[4]); // 50
  10. System.out.println(array[5]); // 50
  11. System.out.println("=================");
  12. // 使用循环,次数其实就是数组的长度。
  13. for (int i = 0; i < 6; i++) {
  14. System.out.println(array[i]);
  15. }
  16. System.out.println("=================");
  17. // int len = array.length; // 长度
  18. for (int i = 0; i < array.length; i++) {
  19. System.out.println(array[i]);
  20. }
  21. }
  22. }

5.5 数组的最值

  1. public class Demo01 {
  2. public static void main(String[] args) {
  3. int[] array1 = {5, 15, 30, 20, 10000};
  4. int max = array1[0];
  5. for(int i = 1;i < array1.length;i++){
  6. if(array1[i]>max){
  7. max = array1[i];
  8. }
  9. }
  10. System.out.println(max);
  11. }
  12. }

5.6 数组元素反转

5.6.1 一种思路

  1. public static void main(String[] args) {
  2. int[] array1 = {5, 15, 30, 20, 10000};
  3. for(int i = 1;i <= array1.length/2;i++){
  4. int t;
  5. t = array1[array1.length-i];
  6. array1[array1.length-i] = array1[i-1];
  7. array1[i-1] = t;
  8. }
  9. for (int j = 0; j < array1.length; j++) {
  10. System.out.println(array1[j]);
  11. }
  12. }

5.6.2 另一种思路

数组 - 图5

  1. int[] array2 = {5, 15, 30, 20, 10000};
  2. for(int min = 0, max = array2.length - 1; min < max; min++, max--){
  3. int t = array2[min];
  4. array2[min] = array2[max];
  5. array2[max] = t;
  6. }
  7. for (int j = 0; j < array2.length; j++) {
  8. System.out.println(array2[j]);
  9. }

六、数组作为方法的参数和返回值

  • 数组作为参数时,传的是数组的地址值,所以在方法内修改数组时,原数组也会发生相应修改。

  • 数组作为返回值时,传的也是数组的地址值。