第一章 数组的定义和访问
1.1 数组的概念及特点
概念
是一种容器,可以同时存放多个数据值
特点
1,数组是一种引用数据类型
2,数组当中的多个数据,类型必须统一
3,数组的长度在数据运行期间不可改变
1.2 数组的初始化
概念
即在内存中创建一个数组,并且向其中赋予一些默认值
初始化方式
两种常见方式
1,动态初始化(指定长度)
2,静态初始化(指定内容)
动态初始化数组的格式
数据类型 [] 数组名称 = new 数据类型 [数组长度];
静态初始化基本格式
数据类型[] 数组名称 = new 数据类型[] {元素A, 元素B, 元素C,...};
解释含义
- 左侧数据类型:数组中存放的数据的数据类型
- 左侧的中括号:代表我是一个数组
- 左侧数组名称:给数组去一个名字
- 右侧的new:代表创建数组的动作
- 右侧数据类型:必须和左侧数据类型一致
- 右侧中括号的长度:即数组当中,可以存放的数据个数,是一个int类型数字
public class Demo01Array {
public static void main(String[] args) {
// 创建一个数组,里面可以存放200个int数据
// 数据类型 [] 数组名称 = new 数据类型 [数组长度];
int[] arrayA = new int[200];
// 创建一个数组,里面可以存放10个doyble数据
double[] arrayB = new double[10];
// 创建一个数组,里面可以存放20个字符串
String[] arrayC = new String[20];
}
}
两者区别
动态初始化(指定长度):在创建数组的时候,直接指定数组当中的数据元素个数
静态初始化(指定内容):在创建数组的时候,不直接指定数组个数的多少,而是将具体的数据内容进行指定
注意事项
虽然静态初始化没有直接告诉长度,但是根据大括号中元素的具体内容,也可以自动推算出长度
public class Demo02Array {
public static void main(String[] args) {
// 创建一个数组,里面装的全是int数字,具体为:5,15, 25
int[] ArrayA = new int[] { 5, 15, 25 };
// 创建一个数组,里面装的全是字符串,具体为:“Hello”,“World”,“java”
String[] ArrayB = new String[] { "Hello", "World", "Java" };
}
}
静态初始化的省略格式
// 标准格式:
数据类型[] 数组名称 = new 数据类型[] {元素A, 元素B, 元素C,...};
// 省略格式:
数据类型[] 数组名称 = {元素A, 元素B, 元素C,...};
注意事项
1,静态初始化没有直接指定长度,但是仍然会自动推算得到长度
2,静态初始化标准格式可以拆分成两个步骤
3,动态初始化也可以拆分成两个步骤
4,静态初始化一旦采用省略格式,就不能拆分成两个步骤
使用建议:
如果不确定数组当中的具体内容,用动态初始化;否则,已经确定了具体的内容,用静态初始化
public class Demo03Array {
public static void main(String[] args) {
// 省略格式静态初始化
int[] arrayA = { 10, 20, 30 };
// 静态初始化的标准格式,可以拆分成两个部分
int[] arrayB;
arrayB = new int[] { 11, 21, 31 };
// 动态初始化也可以拆分成两个部分
int[] arrayC;
arrayC = new int[3];
// 静态初始化的省略部分,不可以拆分成两个步骤
// int[] arrayD;
// arrayD = { 10, 20, 30 };
}
}
1.3 数组的访问
直接打印
直接打印数组名称,得到的是数组对应的:内存地址哈希
-
- [ —> 代表他是个数组
- I —> 代表数组中数据类型为int
- 50cbc42f —> 十六进制地址值
正确访问格式
访问数组元素的格式:数组名(索引值);
索引值:就是一个int数字,代表数组当中元素编号
【注意】索引值从0开始,一直到“数组的长度 - 1”为止
public class Demo04ArrayUse {
public static void main(String[] args) {
// 静态初始化省略格式
int[] array1 = { 10, 20, 30 };
System.out.println(array1); // [I@50cbc42f
// 直接打印数组当中元素
System.out.println(array1[0]); // 10
System.out.println(array1[1]); // 20
System.out.println(array1[2]); // 30
System.out.println("=====================");
// 循环打印数组当中元素
for (int i = 0; i < 3; i++) {
System.out.println(array1[i]);
}
System.out.println("=====================");
// 也可以将数组当中某一个单一元素,赋值交给变量
int sum = array1[1];
System.out.println(sum); //20
}
}
1.4 数组的默认值
默认值
使用动态初始化数组的时候,其中元素将会自动拥有一个默认值。规则如下:
- 如果是整数类型,那么默认为0;
- 如果是浮点类型,那么默认为0.0;
- 如果是字符类型,那么默认为’\u0000’;
- 如果是布尔类型,那么默认为false;
- 如果是引用类型,那么默认为null。
注意事项
- 静态初始化其实也有默认值的过程,只不过系统马上将默认值替换成为大括号当中的具体数值
public class Demo05ArrayUse {
public static void main(String[] args) {
// 动态初始化一个地址
int[] array1 = new int[3];
System.out.println(array1); // 内存地址值
System.out.println("整数类型默认值");
System.out.println(array1[0]); // 0
System.out.println(array1[1]); // 0
System.out.println(array1[2]); // 0
System.out.println("=============");
// 将数据123赋值交给数组array1中的1号元素
array1[1] = 123;
System.out.println(array1[0]); // 0
System.out.println(array1[1]); // 0
System.out.println(array1[2]); // 0
System.out.println("=============");
double[] array2 = new double[1];
System.out.println("浮点类型默认值");
System.out.println(array2[0]); // 0.0
System.out.println("=============");
char[] array3 = new char[1];
System.out.println("字符类型默认值");
System.out.println(array3[0]); // ’/u0000‘
System.out.println("=============");
boolean[] array4 = new boolean[1];
System.out.println("布尔类型默认值");
System.out.println(array4[0]); // false
System.out.println("=============");
System[] array5 = new System[1];
System.out.println("整数类型默认值");
System.out.println(array5[0]); // null
}
}
第二章数组原理内存图
2.1内存概述
内存是计算机中的重要原件,临时存储区域,作用是运行程序。我们编写的程序是存放在硬盘中的,在硬盘中的程 序是不会运行的,必须放进内存中才能运行,运行完毕后会清空内存。 Java虚拟机要运行程序,必须要对内存进行空间的分配和管理。
2.2 Java虚拟机的内存划分
Java的内存需费划分成为5个部分:
1.栈(Stack) : 存放的都是方法中的局部变量。方法的运行一定要在栈当中运行
- 局部变量:方法的参数,或者是方法0内部的变量
- 作用域:一旦超出作用域,立刻从栈内存当中消失。
2.堆(Heap) : 凡是new出来的东西,都在堆当中。
- 堆内存里面的东西都有一个地址值:16进制
堆内存里面的数据,都有默认值。规则:
- 如果是整数 默认为0
- 如果是浮点数 默认为0.0
- 如果是字符 默认为’\u00oo’
- 如果是布尔 默认为false
- 如果是引用类型 默认为null
3.方法区(Method Area) : 存储.class相关信息,包含方法的信息。
4.本地方法栈(Native Method Stack) : 与操作系统相关。
5.寄存器(pc Register) : 与CPU相关。
2.3数组在内存中的存储
一个数组内存图
栈:存储局部变量
堆:存储new出来的东西
方法区:存储.class相关信息
流程
程序想要运行必须要一个.class文件,不能直接运行.java文件,必须需要.class文件,Demo01ArrayOne.class中包含main方法,java会将main方法的相关信息(名字、参数,返回值)保存至方法区
运行时会先查看方法区是否有main方法,有则将他的信息加载到栈,即为main方法在栈中开辟一块内存空间,这本步骤称为入栈
运行时发现在main方法中创建了一个局部变量——数组array,并对其进行了赋值;由于赋值方式为new,会将数组存放到堆中,在堆中开辟一块内存,存放数组的值的相关信息(地址值(16进制,0x代表16进制的值),索引值,数值或默认值),地址值将会被赋值到栈中,放在所属局部变量的后面
注意:new出来的数组实际时存放在堆中,栈中的局部变量存放的只是一个16进制的地址值,
但我们可以通过地址值找到这个数组,比如 System.out.println(array[0]);
第一步:通过局部变量array中的地址值找到数组
第二步:通过索引值”0“就可以找到对应的值,并对其进行修改或赋值
最后,数组的数值会改变,但地址值不会改变
重新运行程序时,地址值可能会改变
public static void main(String[] args) {
int[] array = new int[3]; // 动态地址初始化
System.out.println(array); // 地址值
System.out.println(array[0]); // 0
System.out.println(array[1]); // 0
System.out.println(array[2]); // 0
System.out.println("================");
array[1] = 10;
array[2] = 20;
System.out.println(array); // 地址值
System.out.println(array[0]); // 0
System.out.println(array[1]); // 10
System.out.println(array[2]); // 20
}
两个数组内存图
两个数组之间不会相互影响
public class Demo02ArrayTwo {
public static void main(String[] args) {
int[] arrayA = new int[3]; // 动态地址初始化
System.out.println(arrayA); // 地址值
System.out.println(arrayA[0]); // 0
System.out.println(arrayA[1]); // 0
System.out.println(arrayA[2]); // 0
System.out.println("================");
arrayA[1] = 10;
arrayA[2] = 20;
System.out.println(arrayA); // 地址值
System.out.println(arrayA[0]); // 0
System.out.println(arrayA[1]); // 10
System.out.println(arrayA[2]); // 20
int[] arrayB = new int[3]; // 动态地址初始化
System.out.println(arrayB); // 地址值
System.out.println(arrayB[0]); // 0
System.out.println(arrayB[1]); // 0
System.out.println(arrayB[2]); // 0
System.out.println("================");
arrayB[1] = 100;
arrayB[2] = 200;
System.out.println(arrayB); // 地址值
System.out.println(arrayB[0]); // 0
System.out.println(arrayB[1]); // 100
System.out.println(arrayB[2]); // 200
}
}
两个变量指向一个数组
先创建了局部变量arrayA,然后将arrayA的地址值交给arrayB,arrayB对其中的数值进行修改,
由于array A和arrayB共用一个地址值,所以arrayA的数值也发生了改变
public class Demo03ArraySame {
public static void main(String[] args) {
int[] arrayA = new int[3]; // 动态地址初始化
System.out.println(arrayA); // 地址值
System.out.println(arrayA[0]); // 0
System.out.println(arrayA[1]); // 0
System.out.println(arrayA[2]); // 0
System.out.println("================");
arrayA[1] = 10;
arrayA[2] = 20;
System.out.println(arrayA); // 地址值
System.out.println(arrayA[0]); // 0
System.out.println(arrayA[1]); // 10
System.out.println(arrayA[2]); // 20
System.out.println("================");
int[] arrayB = arrayA; // 将arrayA的地址值赋值给arrayB
System.out.println(arrayB); // 地址值
System.out.println(arrayB[0]); // 0
System.out.println(arrayB[1]); // 10
System.out.println(arrayB[2]); // 20
System.out.println("================");
arrayB[1] = 100;
arrayB[2] = 200;
System.out.println(arrayB); // 地址值
System.out.println(arrayB[0]); // 0
System.out.println(arrayB[1]); // 100
System.out.println(arrayB[2]); // 200
System.out.println("================");
System.out.println(arrayA); // 地址值
System.out.println(arrayA[0]); // 0
System.out.println(arrayA[1]); // 100
System.out.println(arrayA[2]); // 200
}
}
第三章数组的常见操作
3.1数组越界异常
索引:
数组的索引编号从0开始,一直到“数组的长度-1”为止
概述
如果访问数组元素的时候,索引编号不存在,那么将会发生
数组索引越界异常(ArrayIndexOutOfBoundsException)
原因:
索引编号写错了
解决:
修改成为正确索引编号
public class Demo01ArrayIndex {
public static void main(String[] args) {
int[] arrayA = {15, 25, 35};
System.out.println(arrayA[0]); // 15
System.out.println(arrayA[1]); // 25
System.out.println(arrayA[2]); // 35
// 错误写法,数组索引编号不存在
// System.out.println(arrayA[3]); // 35
// System.out.println(arrayA[-1]); // 35
}
}
3.2数组空指针异常
所用引用类型变量,都可以赋值为一个null值。但是代表其中什么也没有
数组必须进行new初始化才能使用其中元素,如果只是赋值了一个null,没有进行new创建数值,
直接访问其中数值,那么将会发生:
空指针异常(NullPointerException)
原因:忘了new
解决:补上new
public class Demo02ArrayNull {
public static void main(String[] args) {
int[] array;
// array = new int[3];
array = null;
System.out.println(array);
// System.out.println(array[0]);
}
}
3.3数组的长度
读取数组长度,格式:
**数组名称.length**<br />
这将会得到一个int数字,代表数组的长度
注意:
**数组一旦创建,程序运行期间,长度不可改变**
程序中arrayC数组两次输出的长度、地址值不一样是因为,一次将有3个元素的数组的地址值为 [I@50cbc42f 的地址值交给arrayC,
一次将有5个元素的数组的地址值为 [I@75412c2f的地址值交给arrayC。名字没变,但地址值发生了改变
public class Demo03ArrayLength {
public static void main(String[] args) {
int[] arrayA = new int[3];
System.out.println(arrayA.length); // 3
int[] arrayB = { 1, 3, 4, 4, 5, 555, 6632, 34, 434, 24, 234, 5, 67, 234, 55 };
System.out.println(arrayB.length); // 15
int[] arrayC = new int[3];
System.out.println(arrayC); // [I@50cbc42f
System.out.println(arrayC.length); // 3
arrayC = new int[5];
System.out.println(arrayC); // [I@75412c2f
System.out.println(arrayC.length); // 5
}
}
3.4数组遍历【重点】
遍历数组:
说的就是对数组中的每一个元素进行逐一,挨个处理。默认的方式就是打印输出
idea偷懒写法—— "数组名.fori" 然后回车
public class Demo04Array {
public static void main(String[] args) {
int[] arrayA = {12, 23, 34, 45, 56, 67, 78, 87};
// 首先使用原始方式
System.out.println(arrayA[0]); // 12
System.out.println(arrayA[1]); // 23
System.out.println(arrayA[2]); // 34
System.out.println(arrayA[3]); // 45
System.out.println(arrayA[4]); // 56
System.out.println(arrayA[5]); // 67
System.out.println(arrayA[6]); // 78
System.out.println("===================");
// 使用for循环,次数为数组的长度 8.fori
for (int i = 0; i < 7; i++) {
System.out.println(arrayA[i]);
}
System.out.println("===================");
// 使用for循环,次数为数组的长度 arrayA.fori
for (int i = 0; i < arrayA.length; i++) {
System.out.println(arrayA[i]);
}
}
}
3.5数组获取最大值元素
for循环中使用if进行判断,获取最大值元素
package JavaRunMen.Java05.demo03;
public class Demo05ArrayMax {
public static void main(String[] args) {
int[] arrayB = {7, 67, 31, 84, 32, 15, 5, 18, 82, 78, 50, 30, 68, 11, 34, 45, 12, 25, 71, 62};
int max =arrayB[0];
for (int i = 0; i < arrayB.length; i++) {
if (arrayB[i] > max) {
max = arrayB[i];
}
}
System.out.println("arrayB数组的最大值是"+max);
}
}
3.6数组反转
package JavaRunMen.Java05.demo03;
/*
* 数组元素反转:
* 本来的样子:[1, 2, 3, 4]
* 之后的样子:[4, 3, 2, 1]
*
* 要求:不能使用新数组,就用原来唯一一个数组
* */
public class Demo07ArrayReverse {
public static void main(String[] args){
int[] arrayA = {2, 8, 10, 11, 26};
for (int i = 0; i < arrayA.length; i++) {
System.out.println(arrayA[i]);
}
/*
* 初始化语句:int min = 0, max = arrayA.length-1
* 条件判断语句:min < max
* 步进表达式:min++, min--
* 循环体:用第三个变量倒手
* */
for (int min = 0, max = arrayA.length-1; min < max; min++, max--) {
int temp = arrayA[min];
arrayA[min] = arrayA[max];
arrayA[max] = temp;
}
System.out.println("====================================");
for (int i = 0; i < arrayA.length; i++) {
System.out.println(arrayA[i]);
}
// 自己写的
// int[] arrayA = {2, 8, 10, 11, 25, 26};
// for (int i = 0; i < arrayA.length/2; i++) {
// int temp = arrayA[i];
// arrayA[i] = arrayA[arrayA.length - 1 - i];
// arrayA[arrayA.length - 1 - i] = temp;
// }
// for (int i = 0; i < arrayA.length; i++) {
// System.out.println(arrayA[i]);
// }
}
}
第四章数组作为方法参数和返回值
4.1数组作为方法参数
数组可以作为方法的参数
当调用方法的时候,向方法的小括号进行传参,传递进去的其实是数组的地址值
package JavaRunMen.Java05.Demo04;
/*
* 数组可以作为方法的参数
* 当调用方法的时候,向方法的小括号进行传参,传递进去的其实是数组的地址值
* 然后方法会根据地址值进行相应的操作
*
* */
public class Demo01ArrayParam {
public static void main(String[] args) {
int[] arrayA = {1, 4, 5, 7, 12};
System.out.println(arrayA); // 地址值
printArray(arrayA); // 传递进去的是arrayA当中保存的地址值
System.out.println("===========AAA===========");
printArray(arrayA);
System.out.println("===========BBB===========");
printArray(arrayA);
}
/*
* 三要素
* 返回值类型:只是进行打印而已,不需要进行计算,也没有结果,用void
* 方法名称:printArray
* 参数列表:必须给我数组,才能打印其中元素。int[] array
* */
public static void printArray(int[] array) {
System.out.println("printArray方法收到的参数是");
System.out.println(array);
for (int i = 0; i < array.length; i++) {
System.out.println(array[i]);
}
}
}
4.2数组作为方法返回值
使用一个数组作为返回值类型,可以返回多个数据结果
数组作为方法的参数,传递进去的其实是数组的地址值。
数组作为方法的返回值,返回的其实也是数组的地址值。
package JavaRunMen.Java05.Demo04;
/*
* 一个方法可以有0、1、2、多个参数,但只能有0或者1个返回值,不能有多个返回值
* 如果希望一个方法当中产生多个结果数据进行返回,怎么办?
* 解决方案:使用一个数组作为返回值类型即可
*
* 任何数据类型都能作为方法的参数类型、或者返回类型
*
* 数组作为方法的参数,传递进去的其实是数组的地址值。
* 数组作为方法的返回值,返回的其实也是数组的地址值。
* */
public class Demo02ArrayReturn {
public static void main(String[] args) {
int[] result = calculate(10, 20,30);
System.out.println("main方法接受到的数组是");
System.out.println(result); // 地址值
System.out.println("总和:" + result[0]);
System.out.println("平均数:" + result[1]);
}
public static int[] calculate (int a, int b, int c) {
int sum = a + b +c; // 总和
int avg = sum / 3; // 平均数
// 两个结果希望进行返回
// 需要一个数组,也就是一个塑料袋,数组可以保存多个结果
/*
* int[] array = new int[2];
* array[0] = sum // 总和
* array[1] = avg // 平均数
*
* */
int[] array = {sum, avg};
System.out.println("calculate方法内部数组是");
System.out.println(array); // 地址值
return array;
}
}
4.3方法的参数类型区别
方法的参数为基本类型时,传递的是数据值. 方法的参数为引用类型时,传递的是地址值
package JavaRunMen.Java05.Demo04;
public class Demo03ArrayParameter {
public static void main(String[] args) {
int a = 1, b = 2;
System.out.println("向方法中传入基本类型的数据");
System.out.println("传入前");
System.out.println(a);
System.out.println(b);
System.out.println("===========================");
System.out.println("方法中");
change(a,b);
System.out.println("===========================");
System.out.println("传入后");
System.out.println(a);
System.out.println(b);
System.out.println("===========================");
int [] arrayA = {1, 3, 5};
System.out.println("向方法中传入引用类型的数据");
System.out.println("传入前");
System.out.println(arrayA[0]);
System.out.println("===========================");
System.out.println("方法中");
change1(arrayA);
System.out.println("===========================");
System.out.println("传入后");
System.out.println(arrayA[0]);
System.out.println("===========================");
System.out.println("方法的参数为基本类型时,传递的是数据值. 方法的参数为引用类型时,传递的是地址值");
}
public static void change(int a, int b) {
a = a + b;
b = a + b;
System.out.println(a);
System.out.println(b);
}
public static void change1(int[] array) {
array[0] = 200;
System.out.println(array[0]);
}
}