一、认识数组
先来认识一下什么数组
1. 什么数组
数组就是一个容器,用来存一批同种类型的数据的。
比如:想要存储 20,10,80,60,90 这些数据。 我们可以把代码写成这样
int[] array = {20,10,80,60,90};
比如:想要存储 “牛二“,“西门“,“全蛋“ 这些数据。我们可以把代码写成这样
String[] names = {"牛二", "西门", "全蛋"};
2. 数组的应用场景
有变量,为什么还要有数组呢? 比如,我们要做一个点名器
如果用变量来做的话,代码是这样子的
如果用数组来做的话,代码是这样子的
一对比我们发现数组的写法比变量的写法更加简洁,所以我们可以得出一个结论
结论:遇到批量数据的存储和操作时,数组比变量更适合
二、数组的定义和访问
各位同学,我们已经知道数组是用来干什么的。那么如何使用Java语言写一个数组呢?这里就需要学习一下数组的初始化格式。
数组有两种初始化的方式,一种是静态初始化、一种是动态初始化。我们先用静态初始化来学习数组的操作。
2.1 数组的静态初始化
所谓静态初始化指的是:在定义数组时直接给数组中的数据赋值。
1. 静态初始化标准格式:
数据类型[] 变量名 = new 数据类型[]{元素1,元素2,元素3};
按照格式定义int类型、double类型数组
//定义数组,用来存储多个年龄 int[] ages = new int[]{12, 24, 36} //定义数组,用来存储多个成绩 double[] scores = new double[]{89.9, 99.5, 59.5, 88.0};
2. 静态初始化简化格式
Java语言的设计者为了简化定义数组的写法,还为静态初始化提供了一种简化写法
数据类型[] 变量名 = {元素1,元素2,元素3};
使用简化格式定义int类型、double类型数组
//定义数组,用来存储多个年龄 int[] ages = {12, 24, 36} //定义数组,用来存储多个成绩 double[] scores = {89.9, 99.5, 59.5, 88.0};
3. 注意哟!!
- 定义数组时,
数据类型[] 数组名
也可写成数据类型 数组名[]
//以下两种写法是等价的。但是建议大家用第一种,因为这种写法更加普遍 int[] ages = {12, 24, 36}; int ages[] = {12, 24, 36}
4. 数组在计算机中的基本原理
我们知道数组是怎么定义的之后,那么接下来看一下数组在计算机中的基本原理。
我们以int[] ages = {12,24,36};
这句话为例,看一下这句话到底在计算机中做了那些事情。
- 首先,左边
int[] ages
表示定义了一个数组类型的变量,变量名叫ages - 其次,右边
{12,24,36}
表示创建一个数组对象,你完全可以把它理解成一个能装数据的东西。这个对象在内存中会有一个地址值[I@4c873330
,每次创建一个数组对象都会有不用的地址值。 - 然后,把右边的地址值
[I@4c873330
赋值给左边的ages变量 - 所以,ages变量就可以通过地址值,找到数组这个东西。
2.2 数组的元素访问
各位同学,通过刚才的学习,我们知道数组是用来存储数据的。那么数组中存储的数据又如何访问呢?这里所说的访问,意思就是获取中数组中数据的值、或者给数组中的数据赋值。
这里先给大家统一几个概念,数组中存储的数据我们叫做元素;而且数组中的每一个元素都有一个编号与之对应,我们把这个编号叫做索引,这个索引是从0依次递增的整数。如下图所示
要想访问数组中的元素,格式如下
//数组名可以找到数组对象的地址,再通过索引就可以定位到具体的元素了 数组名[索引]
接下来用代码来演示一下
//索引: 0 1 2 int[] arr = {12, 24, 36}; // 1、访问数组的全部数据 System.out.println(arr[0]); //12 System.out.println(arr[1]); //24 System.out.println(arr[2]); //36 //下面代码没有3索引,会出现ArrayIndexOutOfBoundsException 索引越界异常 //System.out.println(arr[3]); // 2、修改数组中的数据 arr[0] = 66; arr[2] = 100; System.out.println(arr[0]); //66 System.out.println(arr[1]); 0 System.out.println(arr[2]); //100
除了访问数组中的元素,我们可以获取数组中元素的个数,后面我们统称为数组的长度。
// 3、访问数组的元素个数:数组名.length System.out.println(arr.length); // 技巧:获取数组的最大索引: arr.length - 1(前提是数组中存在数据) System.out.println(arr.length - 1); int[] arr2 = {}; System.out.println(arr2.length - 1);
2.3 数组的遍历
各位同学,接下来我们学习一个对数组最最最常见的操作——数组遍历。所谓遍历意思就是将数组中的元素一个一个的取出来。
我们刚才学习了数组中元素的访问,访问元素必须用到索引,如下列代码。
int[] ages = {12, 24, 36}; System.out.println(ages[0]); System.out.println(ages[1]); System.out.println(ages[2]);
但是,如果数组中有很多很多元素,索引靠自己一个一个数肯定是不行的!我们可以使用for循环从0开始一直遍历到长度-1的位置,就可以获取所有的索引了。
当你获取到每一个索引,那么每一个元素不就获取到了吗?上代码吧
int[] ages = {12, 24, 36}; for (int i = 0; i < ages.length; i++) { // i的取值 = 0,1,2 System.out.println(ages[i]); }
2.4 数组静态初始化案例
学习完数组的静态初始化之后,接下来我们做一个练习题来巩固一下。
需求:某部门5名员工的销售额分别是:16、26、36、6、100,请计算出他们部门的总销售额。 需求分析: 1.看到有16、26、36、6、100这5个数据数据,而且数据值很明确; 1)想到,可以使用数组静态初始化把这5个数据存起来 2.请计算出他们部门的总销售额(这不就是求数组中数据的和吗?) 2)必须先将数组中所有的元素遍历出来 3)想要求和,得先有一个求和变量sum 4)再将每一个元素和求和变量sum进行累加(求和思想)
按照分析的思路来写代码
// 1、定义一个数组存储5名员工的销售额 //索引 0 1 2 3 4 int[] money = {16, 26, 36, 6, 100}; // 3、定义一个变量用于累加求和 int sum = 0; // 2、遍历这个数组中的每个数据。 for (int i = 0; i < money.length; i++) { // i = 0 1 2 3 4 sum += money[i]; } System.out.println("员工的销售总额:" + sum);
2.5 数组的动态初始化
各位同学,刚才我们初始化数组时,都是直接将元素写出来。但是还有另一个初始化数组的方式叫 动态初始化。
动态初始化不需要我们写出具体的元素,而是指定元素类型和长度就行。格式如下
//数据类型[] 数组名 = new 数据类型[长度]; int[] arr = new int[3];
下面是动态初始化数组的原理图。我们发现int[] arr
其实就是一个变量,它记录了数组对象的地址值,而且数组中的元素默认值是0。
注意:
使用动态初始化定义数组时,根据元素类型不同,默认值也有所不同。
关于数组动态初始化的格式和原理,咱们就先学习到这里。
2.6 数组动态初始化案例
各位同学,接下来我们做一个数组动态初始化的案例。
案例需求: 某歌唱比赛,需要开发一个系统:可以录入6名评委的打分,录入完毕后立即输出平均分做 选手得分 需求分析: 1.需要录入6名评委的分数,可以用一个数组来保存。 因为在评委没有录入分数之前,还不确定数组中应该存哪些数据。 所以可以使用数组的动态初始化 2.遍历数组中的每一个位置,并录入分数,将分数存入数组中 3.遍历数组中的每一个元素,对元素求和
代码如下
// 1、定义一个动态初始化的数组,负责后期存储6个评委的打分。 double[] scores = new double[6]; Scanner sc = new Scanner(System.in); // 2、遍历数组中的每个位置,录入评委的分数,存入到数组中去 for (int i = 0; i < scores.length; i++) { // i = 0 1 2 3 4 5 System.out.println("请您输入当前第" + (i + 1) +"个评委的分数:"); double score = sc.nextDouble(); scores[i] = score; } // 3、遍历数组中的每个元素进行求和 double sum = 0; for (int i = 0; i < scores.length; i++) { sum += scores[i]; } System.out.println("选手最终得分是:" + sum / scores.length);
三、数组在计算机中的执行原理
好的各位同学,在前面我们已经学习了数组的基本使用,也理解了数组的基本原理。由于数组是一个容器,变量也是一个容器,在理解他们执行原理的时候,有些同学就容易搞混,现在我把他们放在一起带着大家回顾一下他们的会执行原理,顺便带着大家详细理解一下Java程序的执行的内存原理。
3.1 数组的执行原理,Java程序的执行原理
我们以下面的代码,来讲解变量、数组的执原理。
public class ArrayDemo1 { public static void main(String[] args) { int a = 10; System.out.println(a); int[] arr = new int[]{11, 22, 33}; System.out.println(arr); System.out.println(arr[1]); arr[0] = 44; arr[1] = 55; arr[2] = 66; System.out.println(arr[0]); System.out.println(arr[1]); System.out.println(arr[2]); } }
前面我们给大家讲过,程序是在内存中执行的。实际上Java程序是把编译后的字节码加载到Java虚拟机中执行的。
Java为了便于虚拟机执行Java程序,将虚拟机的内存划分为 方法区、栈、堆、本地方法栈、寄存器 这5块区域。同学们需要重点关注的是 方法区、栈、堆。
下面把每一个块内存区域作用介绍一下,我们大致只需要知道每一部分存储什么内容就行。
- 方法区:字节码文件先加载到这里
- 栈:方法运行时所进入的内存区域,由于变量在方法中,所以变量也在这一块区域中
- 堆:存储new出来的东西,并分配地址。由于数组是new 出来的,所以数组也在这块区域。
下面是上面案例执行的内存原理如下图所示,按照① ② ③ ④ ⑤ ⑥ 的标记的顺序来看
总结一下int a = 10
与 int[] arr = new int[]{11,22,33}的区别
- a是一个变量,在栈内存中,a变量中存储的数据就是10这个值。
- arr也是一个变量,在栈中,存储的是数组对象在堆内存中的地址值
// 这里的int a是一个基本类型变量,存储的是一个数值 int a = 10 ; //这里的int[] arr是一个引用类型的变量,存储的是一个地址值 int[] arr = new int[]{44,55,66};
3.2 多个变量指向同一个数组的问题
各位同学,我们了解了数组在内存中的执行原理。我们知道数组类型的变量,指向的是堆内存中数组对象的地址。但是在实际开发中可能存在一种特殊情况,就是多个变量指向同一个数组对象的形式。
讲解这个知识点的目的,是让同学们注意多个变量指向同一个数组对象存在什么问题?
我们先看一段代码
public class ArrayDemo2 { public static void main(String[] args) { // 目标:认识多个变量指向同一个数组对象的形式,并掌握其注意事项。 int[] arr1 = {11, 22, 33}; // 把int类型的数组变量arr1赋值给int类型的数组变量arr2 int[] arr2 = arr1; System.out.println(arr1); System.out.println(arr2); arr2[1] = 99; System.out.println(arr1[1]); arr2 = null; // 拿到的数组变量中存储的值是null System.out.println(arr2); //System.out.println(arr2[0]); //System.out.println(arr2.length); } }
最新Java基础系列课程--Day04-数组(二)https://developer.aliyun.com/article/1423491