十三、数组

1.数组概述

数字就是用来存储一批同种类型数据的内存区域(可以理解为容器)。


内存

10

20

60

80

90


例子:

int [] arr = {20, 10, 80, 60, 90};

String[] names = {"牛二", "西门", "全蛋"};

场景:假如用变量存储这些名字,然后完成随机点名功能,怎么实现?存在有什么问题?

// 使用数组完成
String[] names = {"章鱼哥", "海绵宝宝", "派大星"};

关于数组需要学会什么?

  1. 怎么定义数组存储数据?
  2. 怎么操作数组元素?
  3. 怎么解决实际问题?
  4. 数组内存原理?
  5. 数组使用的注意点?

2.数组的定义

a.静态初始化数组

Ⅰ.定义数组

定义数组的时候直接给数组赋值。

格式:

// 完整格式
数据类型[] 数组名 = new 数据类型[]{元素1, 元素2, 元素3}

double[] scores = new double[]{89.9, 99.5, 59.5, 88.0};

int[] ages = new int[]{12, 24, 36};

// 简化格式
数据类型[] 数组名 = {元素1, 元素2, 元素3};

int[] ages = {12, 24, 36};

基本原理:数组变量名存储的是数组在内存中的地址,数组是引用类型。

int[] ages = {12, 24, 36};

        System.out.println(ages);
		// ages I@4c873330
		//      | |
        //    int 4c873330 是在内存中的十六进制地址

从左往右赋值。ages记录的是数组的第一个地址。


内存


ages

array

12

24

36


总结:

  1. 数组的静态初始化的写法和特点什么样的?
数据类型[] 数组名 = {元素1, 元素2, 元素3};
int[] ages = {12, 24, 36, 48, 60};
double[] scores = {89.9, 99.5, 59.5};

// 完整格式
数据类型[] 数组名 = new 数据类型[]{元素1, 元素2, 元素3};
int[] ages = new int[]{12, 24, 36, 48, 60};
  1. 数组是属于什么类型,数组变量名中存储的是什么?
  • 引用数据类型,存储的数组在内存中的地址信息。

Ⅱ.访问数组

数组名称

// 取值
System.out.println(arr[0]);

// 赋值
arr[2] = 100;
System.out.println(arr[2]);

数组的长度属性:length

// 获取数组的长度(数组元素的个数)
System.out.println(arr.length);

问题:数组的最大索引可以怎么表示?

// 前提:元素个数大于0
数组名.length - 1

Ⅲ.注意事项

  • "数组类型[] 数组名" 也可以写成 "数据类型 数组名[]"
int[] arr = {30, 40};

int arr[] = {30, 40};
  • 什么类型的数组存放什么类型的数据,否则报错。
// 错误示例
int[] arr = new int[]{30, 40, "派大星"};
  • 数组一旦定义出来,程序执行的过程中,长度、类型就固定了。

b.动态初始化数组

Ⅰ.定义数组

定义数组的时候只确定元素的类型和数组的长度,之后再存入具体数据。

数组的动态初始化格式:

数据类型[] 数组名 = new 数据类型[长度];

int[] arr = new int[3];

// 后赋值
arr[0] = 10;
System.out.println(arr[0]);


        // 目标:学会动态初始化数组的定义和使用
        double[] scores = new double[3];
        // 默认 {0.0, 0.0, 0.0}

        // 赋值
        scores[0] = 99.5;
        System.out.println(scores[0]);
        System.out.println(scores[1]);

        // 默认 null
        String[] names = new String[90];
        names[0] = "章鱼哥";
        names[1] = "海绵宝爸";
        System.out.println(names[0]);
        System.out.println(names[1]);
        System.out.println(names[2]);

Ⅱ.总结

  1. 动态初始化的写法是什么样的?
数据类型[] 数组名 = new 数据类型[长度];

int[] ages = new int[4]
  1. 两种数组定义时的特点和场景有什么区别?
  • 当前已经知道存入的元素值,用静态初始化。
  • 当前还不知道要存入哪些数据,用动态初始化。

Ⅲ.动态初始化数组的元素默认值

元素默认值规则:

数据类型

明细

默认值

基本类型

byte、short、char、int、long

0

float、double

0.0

boolean

false

引用类型

类、接口、数组、String

null

两种初始化的使用场景总结、注意事项说明:

  • 动态初始化:只指定数组长度,后期赋值,适合开始知道数据的数量,但是不确定具体元素值的业务场景。
  • 静态初始化:开始就存入元素值,适合一开始就能确定元素值的业务场景。
  • 两种格式的写法是独立的,不可以混用。

3.数组的遍历

  • 遍历:就是一个一个数据的访问。
  • 为什么要遍历?搜索、数据统计等等都需要用到遍历。
int[] ages = {20, 30, 40, 50};

		// 方式一 快捷键 ages.fori
        for (int i =0; i< ages.length; i++) {
            System.out.println(ages[i]);
        }

		// 方式二 快捷键 ages.for
        for (int age : ages) {
            System.out.println(age);
        }

4.数组的案例

a.数组元素求和

需求:某部门5名员工的销售额分别是:16、26、36、6、100,请计算出他们部门的总销售额。

// 定义销售额
		int[] moneys = {16, 26, 36, 6, 100};

		// 定义总销售额
        int count = 0;

		// 遍历累加 方式一
        for (int money : moneys) {
            count += money;
        }
        System.out.println(count);

		// 遍历累加 方式二
		for (int i; i < moneys.length; i++) {
            count += mongos[i];
        }

b.数组求最高值

分析:

  1. 将颜值数据拿到程序中去,用数组装起来。
int[] faceScores = {15, 9000, 10000, 20000, 9500, -5};
  1. 定义一个变量用于记录最大值,这个变量建议默认存储第一个元素作为参照。
int max = faceScores[0];
  1. 遍历数组的元素,如果该元素大于变量存储的元素,则替换变量存储的值为该元素。
for (int i = 1; i < faceScores.length, i++) {
    if (faceScores[i] > max) {
        // 替换
        max = faceScores[i];
    }
}
  1. 完整代码
// 定义颜值数组
        int[] faceScores = {15, 9000, 10000, 20000, 9500, -5};

        // 定义最大值
        int max = faceScores[0];

        for (int faceScore : faceScores) {
            if (max < faceScore) {
                max = faceScore;
            }
        }
        System.out.println("颜值最高为:" + max);

c.猜数字游戏

开发一个幸运小游戏,游戏规则如下:游戏后台随机生成1-20之间的五个数(无所谓是否重复),然后让大家来猜数字:

  • 未猜中提示:”未猜中“,并继续猜测。
  • 猜中提示:”运气不错,猜中了“,并输出该数据第一个出现的位置,且输出全部5个数据,最终结束本游戏。

分析:

  1. 随机生成5个1-20之间的数据使用数组存储起来。
  2. 定义一个死循环,输入数据猜测,遍历数组,判断数据是否在数组中,如果在,进行对应提示并结束死循环,如果没有猜中,提示绩效猜测直到猜中为止。
// 开发一个幸运小游戏,游戏规则如下:游戏后台随机生成1-20之间的五个数(无所谓是否重复),然后让大家来猜数字:
        //- 未猜中提示:”未猜中“,并继续猜测。
        //- 猜中提示:”运气不错,猜中了“,并输出该数据第一个出现的位置,且输出全部5个数据,最终结束本游戏。

        // 1.定义动态初始化的数组存储5个随机嗯1-20之间的数据
         int[] luckNumbers = new int[5];

         // 2.动态的生成5个1-20之间的随机数并存入到数组中去
        Random rd = new Random();
        for (int i = 0; i < luckNumbers.length; i++) {
            luckNumbers[i] = rd.nextInt(1, 20);
        }

        // 3.获取键盘扫描器对象
        Scanner sc = new Scanner(System.in);

        // 4.死循环开始游戏
        // 设置标号 OUT
        OUT:
        while (true) {
            System.out.println("请你输入一个1-20的整数进行猜测:");

            // 输入数字
            int inputNumber = sc.nextInt();

            // 遍历幸运数组
            for (int luckNumber : luckNumbers) {
                // 判断是否猜中
                if (luckNumber == inputNumber) {
                    System.out.println("运气不错,猜中了, 您猜中的数据是:" + luckNumber);
                    // 通过break OUT 结束整个循环
                    break OUT;
                }
            }

            // 猜错
            System.out.println("未猜中, 请继续猜数字!");
        }

        // 所有幸运数字
        for (int luckNumber : luckNumbers) {
            System.out.print(luckNumber + "\t");
        }

d.随机排名

需求:某公司开发部5名开发人员,要进行项目进展汇报演讲,现在采取随机排名进行汇报。

请先依次录入5名员工的工号,然后展出一组随机的排名顺序。

22 33 35 13 88 -> 13 35 88 33 22

分析:

  1. 在程序中录入5名员工的工号存储起来 -> 使用数组。
  2. 依次遍历数组中的每个元素,随机一个索引数据,让当前元素与该索引位置处的元素进行交换。
package com.javase.arraycase;

import java.util.Random;
import java.util.Scanner;

public class ArrayCaseDemo4 {
    public static void main(String[] args) {
        // 需求:某公司开发部5名开发人员,要进行项目进展汇 报演讲,现在采取随机排名进行汇报。
        // 请先依次录入5名员工的工号,然后展出一组随机的排名顺序。

        // 1.定义动态数组
        int[] numbers = new int[5];

        // 2.循环输入5次员工工号 存入数组中
        Scanner sc = new Scanner(System.in);

        for (int i = 0; i < 5; i++) {
            System.out.println("请您输入第" + (i+1) + "位的员工工号:");

            int number = sc.nextInt();

            numbers[i] = number;
        }

        System.out.println(numbers[0]);

        // 3.随机排序
        Random r = new Random();
        for (int i = 0; i < numbers.length; i++) {
            // 定义随机索引
            int index = r.nextInt(0, 5);

            // 定义交换临时中间变量
            int temp;
            
            temp = numbers[i];
            numbers[i] = numbers[index];
            numbers[index] = temp;
        }

        // 4.展出一组随机的排名顺序
        System.out.println("随机排名:");
        for (int number : numbers) {
            System.out.print(number + "\t");
        }
    }
}

总结:

  1. 如何实现随机排名的?
  • 定义一个动态初始化的数组用于录入数据。
  • 遍历数组中的每个元素,每次随机一个索引值,让当前元素与该索引位置处的元素进行交换。

e.数组排序

就是对数组中的元素,进行升序(由小到大)或者降序(由大到小)的操作。

数组排序的技术

  • 冒泡排序
  • 选择排序
  • 快速排序
  • 插入排序

数据搜索相关的技术

  • 二分搜索
  • 分开查找
  • 哈希表查找

Ⅰ.冒泡排序

冒泡排序的思想s

  • 每次从数组中找出最大值放在数组的后面去。

实现冒泡排序的关键步骤分析

  • 确定总共需要做几轮:数组的长度-1 (四个元素找出三个最大值)。
package com.javase.arraysort;

public class ArraySortDemo1 {
    public static void main(String[] args) {
        // 1.定义一个数组,存储一些数据
        // 索引       0  1  2  3
        int[] arr = {5, 2, 3, 1};

        // 2.定义一个循环控制比较的论数
        for (int i = 0; i < arr.length - 1; i++) {
            // 如果一轮循环完成之后 没有发生交换 说明数组本身就有序 后续循环无需再做
            boolean swapped = false;

            for (int j = 0; j < arr.length - i -1; j++) {
                if (arr[j] > arr[j + 1]) {
                    int temp = arr[j + 1];
                    arr[j + 1] = arr[j];
                    arr[j] = temp;

                    swapped = true;

                }
            }

            // 如果没有发生交换 跳出循环
            if (!swapped) {
                break;
            }
        }

        for (int i : arr) {
            System.out.println(i);
        }
    }
}

总结

  1. 冒泡排序的思想?
  • 从头开始两两比较,把较大的元素与较小的元素进行交换。
  • 每轮把当前最大的一个元素存入到数组当前的末尾。
  1. 冒泡排序的实现步骤?
  • 定义一个外部循环控制总共需要冒几轮(数组的长度-1)
  • 定义一个内部循环,控制每轮依次往后比较几个位置(数组长度-i-1)。
  • 如果当前位置的元素大于后一个位置的元素值,两者交换。

5.数组的内存图

a.Java内存分配、数组内存图

Ⅰ.Java内存分配介绍

  • 栈:方法运行时所进入的内存,变量也是在这里。
  • 堆:new出来的东西会在这块内存中开辟空间并产生地址。
  • 方法区:字节码文件(HelloWorld.class)加载时进入的内存。
  • 本地方法栈
  • 寄存器
package com.javase.ram;

public class RamDemo1 {
    public static void main(String[] args) {
        int a = 12;
        System.out.println(a);

        int[] arr = {11, 22, 33};
        System.out.println(arr);

        arr[0] = 44;
        arr[1] = 55;
        arr[2] = 66;

        System.out.println(arr[0]);
        System.out.println(arr[1]);
        System.out.println(arr[2]);
    }
}


方法区

栈内存

Test.class


方法区main

栈内存main

a

10

栈内存arr

3fee733d

堆内存

堆内存arr

55

66

44-3fee733d


b.两个变量指向同一个数组

package com.javase.ram;

public class RamDemo2 {
    public static void main(String[] args) {
        // 目标:两个变量指向同一个数组
        int[] arr1 = {11, 22, 33};

        // 把数字arr1变量赋值给int类型的数组变量arr2
        int[] arr2 = arr1;

        System.out.println(arr1);
        System.out.println(arr2);

        arr2[1] = 99;
        System.out.println(arr1[1]);

        System.out.println(arr2[0]);
        System.out.println(arr2[1]);
        System.out.println(arr2[2]);
        
    }
}


方法区

栈内存

Test.class


方法区main

栈内存main

栈内存arr1

3fee733d

栈内存arr2

堆内存

堆内存arr

55

66

44-3fee733d


6.数组使用的常见问题

  • 问题一:如果访问的元素位置超过最大索引,执行时会出现ArrayIndexOutOfBoundsException(数组索引越界异常)。
int[] arr = new int[]{11, 22, 33};
		System.out.println(arr[2]);

		// 出现异常
		// System.out.println(arr[3]);
  • 问题二:如果数组变量中没有存储数组的地址,而是null,在访问数组信息时会出现NullPointerException(空指针异常)。
int[] arr = new int[]{11, 22, 33};

		arr = null;

		// 出现异常
        // System.out.println(arr.length);

7.Debug工具的使用

IDEA自带的断点调试(排错)工具,可以控制代码从断点开始一行一行的执行,然后详细观看程序执行的情况。

DEBUG工具基本使用步骤:

  1. 在需要控制的代码行左侧,点击一下,形成断电。
  2. 选择使用Debug方式启动程序,启动后程序会在断点暂停。
  3. 控制代码一行一行的往下执行。