数组
本章目标
- 理解容器的概念
- 掌握数组的第一种定义方式
- 掌握数组的第二种定义方式
- 掌握数组的第三种定义方式
- 使用索引访问数组的元素
- 了解数组的内存图解
- 了解空指针和越界异常
- 掌握数组的遍历
- 掌握数组最大值的获取
- 了解数组反转的原理
- 了解数组作为方法参数传递
- 了解数组作为方法的返回值
1. 数组的定义和访问
1.1 容器的概述
案例分析
现在需要统计某公司员工的工资情况,例如计算平均工资、找到最高工资等。假设该公司有50名员工,用前面所学 的知识,程序首先需要声明50个变量来分别记住每位员工的工资,然后在进行操作,这样做会显得很麻烦,而且错 误率也会很高。因此我们可以使用容器进行操作。将所有的数据全部存储到一个容器中,统一操作。
容器概念
- 容器:是将多个数据存储到一起,每个数据称为该容器的元素
- 生活中的容器:水杯,衣柜,教室…
1.2 数组概念
数组概念: 数组就是存储数据长度固定的容器,保证多个数据的数据类型要一致。
1.3 数组的定义
方式一
-
格式:
数组存储的数据类型[] 数组名字 = new 数组存储的数据类型[长度];
-
数组定义格式详解:
- 数组存储的数据类型: 创建的数组容器可以存储什么数据类型。
- [] : 表示数组。 数组名字:为定义的数组起个变量名,满足标识符规范,可以使用名字操作数组。
- new:关键字,创建数组使用的关键字。
- 数组存储的数据类型: 创建的数组容器可以存储什么数据类型。
- [长度]:数组的长度,表示数组容器中可以存储多少个元素。
- 注意:数组有定长特性,长度一旦指定,不可更改。
- 和水杯道理相同,买了一个2升的水杯,总容量就是2升,不能多也不能少
-
举例:
定义可以存储3个整数的数组容器,代码如下:
int[] arr = new int[3];
方式二
-
格式:
数据类型[] 数组名 = new 数据类型[]{元素1,元素2,元素3...};
-
举例:
定义存储1,2,3,4,5整数的数组容器:
int[] arr = new int[]{1,2,3,4,5};
方式三
-
格式:
数据类型[] 数组名 = {元素1,元素2,元素3...};
-
举例:
定义存储1,2,3,4,5整数的数组容器int[] arr = {1,2,3,4,5};
1.4 数组的访问
-
索引: 每一个存储到数组的元素,都会自动的拥有一个编号,从0开始,这个自动编号称为数组索引 (index),可以通过数组的索引访问到数组中的元素。
-
格式:
数组名[索引]
-
数组的长度属性: 每个数组都具有长度,而且是固定的,Java中赋予了数组的一个属性,可以获取到数组的 长度,语句为:
数组名.length
,属性length的执行结果是数组的长度,int类型结果。由次可以推断出,数 组的最大索引值为数组名.length-1
。public static void main(String[] args) { int[] arr = new int[]{1,2,3,4,5}; //打印数组的属性,输出结果是5 System.out.println(arr.length); }
-
索引访问数组中的元素:
- 数组名[索引]=数值,为数组中的元素赋值
- 变量=数组名[索引],获取出数组中的元素
public static void main(String[] args) { //定义存储int类型数组,赋值元素1,2,3,4,5 int[] arr = {1,2,3,4,5}; //为0索引元素赋值为6 arr[0] = 6; //获取数组0索引上的元素 int i = arr[0]; System.out.println(i); //直接输出数组0索引元素 System.out.println(arr[0]); }
2. 数组原理内存图
2.1 内存概述
内存是计算机中的重要原件,临时存储区域,作用是运行程序。我们编写的程序是存放在硬盘中的,在硬盘中的程序是不会运行的,必须放进内存中才能运行,运行完毕后会清空内存。
Java虚拟机要运行程序,必须要对内存进行空间的分配和管理。
2.2 Java虚拟机的内存划分
为了提高运算效率,就对空间进行了不同区域的划分,因为每一片区域都有特定的处理数据方式和内存管理方式。
-
JVM的内存划分:
区域名称 作用 寄存器 给CPU使用,和我们开发无关。 本地方法栈 JVM在使用操作系统功能的时候使用,和我们开发无关 方法区 存储可以运行的class文件。 堆内存 存储对象或者数组,new来创建的,都存储在堆内存。 方法栈 方法运行时使用的内存,比如main方法运行,进入方法栈中执行。 图解如下:
2.3 数组在内存中的存储
-
数组内存图
public static void main(String[] args) { int[] arr = new int[3]; System.out.println(arr);//[I@5f150435 }
以上方法执行,输出的结果是
[I@5f150435
,这个是什么呢?是数组在内存中的地址。new出来的内容,都是在堆 内存中存储的,而方法中的变量arr保存的是数组的地址。输出arr[0],就会输出arr保存的内存地址中数组中0索引上的元素
图解如下:
-
两个变量指向一个数组
public static void main(String[] args) { // 定义数组,存储3个元素 int[] arr = new int[3]; //数组索引进行赋值 arr[0] = 5; arr[1] = 6; arr[2] = 7; //输出3个索引上的元素值 System.out.println(arr[0]); System.out.println(arr[1]); System.out.println(arr[2]); //定义数组变量arr2,将arr的地址赋值给arr2 int[] arr2 = arr; arr2[1] = 9; System.out.println(arr[1]); }
图解内存如下:
3. 数组的常见操作
3.1 数组越界异常
观察一下代码,运行后会出现什么结果。
public static void main(String[] args) {
int[] arr = {1,2,3};
System.out.println(arr[3]);
}
创建数组,赋值3个元素,数组的索引就是0,1,2,没有3索引,因此我们不能访问数组中不存在的索引,程序运 行后,将会抛出 ArrayIndexOutOfBoundsException
数组越界异常。在开发中,数组的越界异常是不能出现的,一 旦出现了,就必须要修改我们编写的代码。
3.2 数组空指针异常
观察一下代码,运行后会出现什么结果。
public static void main(String[] args) {
int[] arr = {1,2,3};
arr = null;
System.out.println(arr[0]);
}
arr = null 这行代码,意味着变量arr将不会在保存数组的内存地址,也就不允许再操作数组了,因此运行的时候会抛出NullPointerException
空指针异常。在开发中,数组的越界异常是不能出现的,一旦出现了,就必须要修改我们编写的代码。
3.3 数组遍历【重点】
-
数组遍历: 就是将数组中的每个元素分别获取出来,就是遍历。遍历也是数组操作中的基石。
public static void main(String[] args) { int[] arr = { 1, 2, 3, 4, 5 }; System.out.println(arr[0]); System.out.println(arr[1]); System.out.println(arr[2]); System.out.println(arr[3]); System.out.println(arr[4]); }
-
以上代码是可以将数组中每个元素全部遍历出来,但是如果数组元素非常多,这种写法肯定不行,因此我们需要改 造成循环的写法。数组的索引是
0
到length-1
,可以作为循环的条件出现public static void main(String[] args) { int[] arr = { 1, 2, 3, 4, 5 }; for (int i = 0; i < arr.length; i++) { System.out.println(arr[i]); } }
-
可使用增强for循环遍历数组,代码如下:
public static void main(String[] args) { int[] arr = { 1, 2, 3, 4, 5 }; for (int i : arr) { System.out.println(i); } }
3.4 数组获取最大值元素
-
最大值获取:从数组的所有元素中找出最大值。
-
实现思路:
- 定义变量,保存数组0索引上的元素
- 遍历数组,获取出数组中的每个元素
- 将遍历到的元素和保存数组0索引上值的变量进行比较
- 如果数组元素的值大于了变量的值,变量记录住新的值
- 数组循环遍历结束,变量保存的就是数组中的最大值
-
示例代码如下:
public static void main(String[] args) { int[] arr = { 5, 15, 2000, 10000, 100, 4000 }; //定义变量,保存数组中0索引的元素 int max = arr[0]; //遍历数组,取出每个元素 for (int i = 0; i < arr.length; i++) { //遍历到的元素和变量max比较 //如果数组元素大于max if (arr[i] > max) { //max记录最大值 max = arr[i]; } } System.out.println("数组最大值是: " + max); }
3.5 数组反转
-
数组的反转: 数组中的元素颠倒顺序,例如原始数组为1,2,3,4,5,反转后的数组为5,4,3,2,1
-
实现思路:数组最远端的元素互换位置。
- 实现反转,就需要将数组最远端元素位置交换
- 定义两个变量,保存数组的最小索引和最大索引
- 两个索引上的元素交换位置 最小索引++,最大索引–,再次交换位置
- 最小索引超过了最大索引,数组反转操作结束
-
示例代码如下:
public static void main(String[] args) { int[] arr = { 1, 2, 3, 4, 5 }; /* 循环中定义变量min=0最小索引 max=arr.length‐1最大索引 min++,max‐‐ */ for (int min = 0, max = arr.length ‐ 1; min <= max; min++, max‐‐) { //利用第三方变量完成数组中的元素交换 int temp = arr[min]; arr[min] = arr[max]; arr[max] = temp; } // 反转后,遍历数组 for (int i = 0; i < arr.length; i++) { System.out.println(arr[i]); } }
4. 数组作为方法参数和返回值
4.1 数组作为方法参数
以前的方法中我们学习了方法的参数和返回值,但是使用的都是基本数据类型。那么作为引用类型的数组能否作为 方法的参数进行传递呢,当然是可以的。
-
数组作为方法参数传递,传递的参数是数组内存的地址。
public static void main(String[] args) { int[] arr = { 1, 3, 5, 7, 9 }; //调用方法,传递数组 printArray(arr); } /* 创建方法,方法接收数组类型的参数 进行数组的遍历 */ public static void printArray(int[] arr) { for (int i = 0; i < arr.length; i++) { System.out.println(arr[i]); } }
4.2 数组作为方法返回值
-
数组作为方法的返回值,返回的是数组的内存地址
public static void main(String[] args) { //调用方法,接收数组的返回值 //接收到的是数组的内存地址 int[] arr = getArray(); for (int i = 0; i < arr.length; i++) { System.out.println(arr[i]); } } /* 创建方法,返回值是数组类型 return返回数组的地址 */ public static int[] getArray() { int[] arr = { 1, 3, 5, 7, 9 }; //返回数组的地址,返回到调用者 return arr; }
图解内存如下:
4.3 方法的参数类型区别
代码分析
-
分析下列程序代码,计算输出结果
public static void main(String[] args) { int a = 1; int b = 2; System.out.println(a); System.out.println(b); change(a, b); System.out.println(a); System.out.println(b); } public static void change(int a, int b) { a = a + b; b = b + a; }
-
分析下列程序代码,计算输出结果。
public static void main(String[] args) { int[] arr = {1,3,5}; System.out.println(arr[0]); change(arr); System.out.println(arr[0]); } public static void change(int[] arr) { arr[0] = 200; }
总结:
方法的参数为基本类型时,传递的是数据值. 方法的参数为引用类型时,传递的是地址值.
5. Arrays类
5.1 概述
java.util.Arrays
类包含用来操作数组的各种方法,比如排序和搜索等。其所有方法均为静态方法,调用起来非常简单。
5.2 操作数组的方法
方法名称 | 说明 |
---|---|
boolean equals(array1,array2) | 比较array1和array2两个数组是否相等 |
sort(array) | 对数组array的元素进行升序排列 |
String toString(array) | 将一个数组array转换成一个字符串 |
void fill(array,val) | 把数组array所有元素都赋值为val |
copyOf(array,length) | 把数组array复制成一个长度为length的新数组,返回类型与复制的数组一致 |
int binarySearch(array, val) | 查询元素值val在数组array中的下标(要求数组中元素已经按升序排列) |
5.3 练习
演示Arrays的sort方法如下:
import java.util.Arrays;
import java.util.Scanner;
/*
* Arrays排序5名学员成绩
* */
public class ArraysSortScore {
public static void main(String[] args) {
int[] scores=new int[5];
Scanner input=new Scanner(System.in);
System.out.println("请输入5名学员的成绩:");
//获取学员成绩
for(int i=0;i<scores.length;i++){
scores[i]=input.nextInt();
}
Arrays.sort(scores);
System.out.print("学员成绩按升序排列:");
for(int score:scores){
System.out.print(score+"\t");
}
System.out.print("\n学员成绩按降序排列:");
for(int i=scores.length-1;i>=0;i--){
System.out.print(scores[i]+"\t");
}
}
}
演示Arrays类各种方法,代码如下:
import java.util.Arrays;
public class ArraysMethod {
public static void main(String[] args) {
//equals(array1,array2) 的作用是:比较两个数组是否相等
System.out.println("----equals(array1,array2)方法:----");
int [] arr1 = {10,50,40,30};
int [] arr2 = {10,50,40,30};
int [] arr3 = {60,50,85};
System.out.println(Arrays.equals(arr1, arr2)); //判断arr1与arr2的长度及元素是否相等
System.out.println(Arrays.equals(arr1, arr3)); //判断arr1与arr3的长度及元素是否相等
//sort(array) 的作用是:对数组array的元素进行升序排列
System.out.println("----sort(array)方法:----");
int [] array = new int[]{80,66,70,54,98};
Arrays.sort(array);
System.out.println("成绩排序后:");
for(int i = 0; i < array.length; i++) {
System.out.println(array[i]);
}
//toString(array) 的作用是:将一个数组array转换成一个字符串
System.out.println("----toString(array)方法:----");
int[] arr = { 10, 50, 40, 30 };
Arrays.sort(arr); //将数组按升序排列
System.out.println(Arrays.toString(arr)); //将数组arr转换为字符串并输出
//fill(array,val) 的作用是:把数组array的所有元素都赋值为val
System.out.println("----fill(array,val)方法:----");
int[] arrs = { 10, 50, 40, 30 }; //初始化整型数组
Arrays.fill(arrs, 40); //替换数组元素
System.out.println(Arrays.toString(arrs)); //将数组arr转换为字符串并输出
//copyOf(array,length) 的作用是:把数组array复制成一个长度为length的新数组
System.out.println("----copyOf(array,length)方法:----");
int[] arr11 = { 10, 50, 40, 30 };
int[] arr22 = Arrays.copyOf(arr11, 3); //将arr11复制成长度为3的新数组arr22
System.out.println(Arrays.toString(arr22));
int[] arr33 = Arrays.copyOf(arr11, 4); //将arr11复制成长度为4的新数组arr33
System.out.println(Arrays.toString(arr33));
int[] arr44 = Arrays.copyOf(arr11, 6); //将arr11复制成长度为6的新数组arr44
System.out.println(Arrays.toString(arr44));
//binarySearch(array, val) 的作用是:查询元素值val在数组array中的下标
System.out.println("----binarySearch(array, val)方法:----");
int[] arrss = { 10, 50, 40, 30 };
Arrays.sort(arrss); //先按升序排列
int index=Arrays.binarySearch(arrss, 30); //查找30的下标
System.out.println(index);
index=Arrays.binarySearch(arrss, 50); //查找50的下标
System.out.println(index);
}
}
使用Arrays类升序排列一组字符,并查找某个特殊字符在升序后数组中的位置
import java.util.Arrays;
/**
* 使用Arrays类升序排列一组字符,并查找某个特殊字符在升序后数组中的位置
* */
public class ArraysSortChar {
public static void main(String[] args) {
// 字符排序
char[] chars = new char[] { 'a', 'c', 'u', 'b', 'e', 'p', 'f', 'z' };
System.out.print("原字符序列:");
for (int i = 0; i < chars.length; i++) {
System.out.print(chars[i] + " ");
}
Arrays.sort(chars); // 对数组进行升序排序
System.out.print("\n升序排序后:");
for (int i = 0; i < chars.length; i++) {
System.out.print(chars[i] + " ");
}
// 查找b的位置
int index=Arrays.binarySearch(chars,'b');
System.out.println("\n'b'在升序后数组中的位置是:"+index);
}
}
6. 二维数组
二维数组就是数组的数组。定义一个二维数组语法如下:
<数据类型> [ ][ ] 数组名;
或者
<数据类型> 数组名 [ ][ ];
定义二维数组时,要定义最大维数
int[ ][ ] scores; //定义二维数组
scores=new int[5][50]; //分配内存空间
//或者
int[ ][ ] scores = new int[5][50];
二维数组实际上是一个以一维数组做为元素的一维数组
二维数组与内存如下:
int[][] s =new int[3][5];
int[][] scores=new int[3][5];
score[0][0]=90;
score[2][3]=70;
二维数组定义并赋值
-
写法一
int[][] scores=new int[][]{ { 90, 85, 92, 78, 54 }, { 76, 63,80 }, { 87 }};
-
写法二
int scores[][] = {{ 90, 85, 92, 78, 54 }, { 76, 63,80 }, { 87 } };
遍历二维数组
有5个班各5名学生某门课程的成绩,计算5个班各自的总成绩,代码如下:
import java.util.Scanner;
/**
* 有5个班各5名学生某门课程的成绩,计算5个班各自的总成绩
* */
public class FiveTotal {
public static void main(String[] args) {
Scanner input=new Scanner(System.in);
int [][] array = new int[5][5]; //5个班的成绩
//i:班级 j:各班级的学生
for(int i=0;i<array.length;i++){
System.out.println("**********第"+(i+1)+"个班**********");
for(int j=0;j<array[i].length;j++){
System.out.print("请输入第"+(j+1)+"个学生的成绩:");
array[i][j]=input.nextInt();
}
}
System.out.println("***********成绩统计************");
int total; //保存总成绩
for(int i = 0; i < array.length; i++) {
String str = (i+1) + "班";
total = 0; //每次循环到此都将其归0
for(int j = 0; j < array[i].length; j++) {
total += array[i][j]; //成绩叠加
}
System.out.println(str+"总成绩:" + total);
}
}
}
已知有3个班级各5名学员,请使用二维数组计算各个班级的总成绩,代码如下:
import java.util.Scanner;
/**
* 已知有3个班级各5名学员,请使用二维数组计算各个班级的总成绩
* */
public class ThreeTotal {
public static void main(String[] args) {
Scanner input=new Scanner(System.in);
int [][] array = new int[3][5]; //3个班的成绩
//i:班级 j:各班级的学生
for(int i=0;i<array.length;i++){
System.out.println("**********第"+(i+1)+"个班**********");
for(int j=0;j< array[i].length;j++){
System.out.print("请输入第"+(j+1)+"个学生的成绩:");
array[i][j]=input.nextInt();
}
}
System.out.println("***********成绩统计************");
int total; //保存总成绩
for(int i = 0; i < array.length; i++) {
String str = (i+1) + "班";
total = 0; //每次循环到此都将其归0
for(int j = 0; j < array[i].length; j++) {
total += array[i][j]; //成绩叠加
}
System.out.println(str+"总成绩:" + total);
}
}
}
小结
二维数组就是数组的数组,三维数组就是二维数组的数组;
多维数组的每个数组元素长度都不要求相同;
打印多维数组可以使用Arrays.deepToString()
;
最常见的多维数组是二维数组,访问二维数组的一个元素使用array[row][col]
。