数组

数组定义和访问

容器的概念

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
变量中只能存储一个数据,新的数据进来,老的数据将被替换,如果需要存储多个数据,需要使用容器。

容器概念
容器:是将多个数据存储到一起,每个数据称为该容器的元素。
生活中的容器:水杯,衣柜,鞋柜,教室

数组:
1.概念: java中的数组就是一个容器,可以存储多个数据,但是数据的类型必须一致
2.特点:
(1)可以存储多个数据
(2)多个数据的类型必须保持一致
(3)数组一旦创建,长度是永远不可以发生改变
(4)创建数组对象会在内存中开辟一整块连续的空间,而数组名中引用的是这块连续空间的首地址

public class Demo01Box { //-----------------------此代码不用敲--------------------------------
public static void main(String[] args) {
//定义一个int变量num,并初始化
//一个变量中只能存储一个数据
int num = 10;
System.out.println(num);

//新的数据进来,老的数据将被替换
num = 20;
System.out.println(num);


//比如我们班有80个学生需要参加考试,每个学生对应一个成绩(整数)
//可以定义80个int变量
int a = 80;
int b = 90;
int c = 70;
int d = 50;
int e = 85;
int f = 95;
int g = 59;
int h = 63;

//需要求总分
int sum = a + b + c + d + e + f + g + h;

//需要求平均分
int avg = sum/80;

//需要求最高分和最低分: 实现起来非常麻烦


}
}

数组变量的定义

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
1.数组变量定义格式一:			---------------推荐使用-------------
数据类型[] 一维数组名称;
数据类型[][] 二维数组名称;

2.数组变量定义格式二:
数据类型 数组名称[];


1.定义一个存储int类型数组的变量arrayA
2.定义一个存储double类型数组的变量arrayB
3.定义一个存储char类型数组的变量arrayC

public class Demo02Array {
public static void main(String[] args) {
//定义int变量num
int num;
//System.out.println(num);//错误:变量未初始化,不能使用

//(1)定义一个存储int类型数组的变量arrayA
//int类型数组: 该容器中只能存储的是int类型的数字
int[] arrayA;
//System.out.println(arrayA);//错误的: arrayA只是一个用来存储数组的变量,但是目前没有向arrayA中存储数组

//(2)定义一个存储double类型数组的变量arrayB
//double类型数组: 该数组容器中只能存储double类型的数据
double arrayB[];

//(3)定义一个存储char类型数组的变量arrayC
//char类型数组: 该数组容器中只能存储char类型的数据
char[] arrayC;
}
}

图解:

image-20200515100251817

数组的第一种初始化方式

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
1.创建一个int类型的数组,可以存储3int数据,给该数组起个名称叫做arrayA
2.创建一个double类型的数组,可以存储7double数据,给该数组起个名称叫做arrayB
3.创建一个char类型的数组,可以存储5char数据,给该数组起个名称叫做arrayC

数组的第一种初始化方式(动态初始化: 指定数组的长度)
1.格式:
数据类型[] 数组名称 = new 数据类型[长度];

2.格式解释:
(1)左侧数据类型: 规定了数组中可以存储哪种类型的数据
(2)左侧[]: 表示数组的意思
(3)数组名称: 就是给数组起个名字(相当于门牌号),方便找到并使用数组
(4)=: 表示赋值,把具体的=右边new出来的数组容器,
存储到=左边的数组变量中,但是存储的是数组在内存空间的地址值
(5)new: 创建数组容器的过程
(6)右侧数据类型: 和左侧保持一致
(7)右侧[]中长度: 是一个int数字,表示数组中可以存储数据的数量,也叫作数组长度

3.练习:
(1)创建一个int类型的数组,可以存储3int数据,给该数组起个名称叫做arrayA
(2)创建一个double类型的数组,可以存储7double数据,给该数组起个名称叫做arrayB
(3)创建一个char类型的数组,可以存储5char数据,给该数组起个名称叫做arrayC

public class Demo03Array {
public static void main(String[] args) {
//(1)创建一个int类型的数组,可以存储3个int数据,给该数组起个名称叫做arrayA
int[] arrayA = new int[3];

//(2)创建一个double类型的数组,可以存储7个double数据,给该数组起个名称叫做arrayB
double[] arrayB = new double[7];

//(3)创建一个char类型的数组,可以存储5个char数据,给该数组起个名称叫做arrayC
char[] arrayC = new char[5];
}
}

数组的第二种初始化方式

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
数组的第二种初始化方式(标准格式静态初始化:指定数组中的每个元素)
1.格式:
数据类型[] 数组名称 = new 数据类型[]{元素1,元素2,元素3,...元素n};

2.注意:
(1)右侧[]中不能写长度: JVM会根据{}中元素的数量计算/推导出数组的长度
只要右侧[]中写长度,就会报错

(2){}中的每个元素之间有逗号(英文状态)隔开,最后一个元素后面没有逗号

3.练习:
(1)创建一个int类型的数组,可以存储多个int数据100,200,300,给该数组起个名称叫做arrayA
(2)创建一个double类型的数组,可以存储多个double数据1.1,2.2,3.3,4.4,5.5,6.6,7.7,给该数组起个名称叫做arrayB
(3)创建一个char类型的数组,可以存储多个char数据'真','的','好','想','你',给该数组起个名称叫做arrayC

public class Demo04Array {
public static void main(String[] args) {
//(1)创建一个int类型的数组,可以存储多个int数据100,200,300,给该数组起个名称叫做arrayA
int[] arrayA = new int[]{100,200,300};

//(2)创建一个double类型的数组,可以存储多个double数据1.1,2.2,3.3,4.4,5.5,6.6,7.7,给该数组起个名称叫做arrayB
double[] arrayB = new double[]{1.1,2.2,3.3,4.4,5.5,6.6,7.7};

//(3)创建一个char类型的数组,可以存储多个char数据'真','的','好','想','你',给该数组起个名称叫做arrayC
char[] arrayC = new char[]{'真','的','好','想','你'};

}
}

数组的第三种初始化方式

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
数组的第三种初始化方式(简化格式静态初始化:指定数组中的每个元素)
1.格式:
数据类型[] 数组名称 = {元素1,元素2,元素3,...元素n};

2.练习:
(1)创建一个int类型的数组,可以存储多个int数据100,200,300,给该数组起个名称叫做arrayA
(2)创建一个double类型的数组,可以存储多个double数据1.1,2.2,3.3,4.4,5.5,6.6,7.7,给该数组起个名称叫做arrayB
(3)创建一个char类型的数组,可以存储多个char数据'真','的','好','想','你',给该数组起个名称叫做arrayC

3.注意:
(1)右侧不用写长度,JVM根据{}中数组元素的数量推导长度
(2)虽然没有写new,底层仍然有new的过程
(3)动态初始化和标准格式的静态初始化,可以分两步完成
(4)简化格式静态初始化,不能分成两步完成

public class Demo05Array {
public static void main(String[] args) {
//(1)创建一个int类型的数组,可以存储多个int数据100,200,300,给该数组起个名称叫做arrayA
int[] arrayA = {100,200,300};

//(2)创建一个double类型的数组,可以存储多个double数据1.1,2.2,3.3,4.4,5.5,6.6,7.7,给该数组起个名称叫做arrayB
double[] arrayB = {1.1,2.2,3.3,4.4,5.5,6.6,7.7};

//(3)创建一个char类型的数组,可以存储多个char数据'真','的','好','想','你',给该数组起个名称叫做arrayC
char[] arrayC = {'真','的','好','想','你'};

//定义int变量,未赋值
int num;
num = 10;//赋值
System.out.println(num);//10
num = 100;//重新赋值
System.out.println(num);//100
System.out.println("-------------------");
//定义数组变量,未赋值
int[] arr;
arr = new int[3];//动态初始化指定数组长度
arr = new int[]{10,20,30};//标准格式静态初始化

System.out.println("-------------------");
int[] arr2;
//arr2 = {100,200,300};//错误: 简化格式静态初始化,不能分成两步完成

}
}

数组的使用

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
简化格式静态初始化数组的使用
1.数组名称: 代表数组在内存空间的地址值,是一个十六进制的整数数字
2.索引编号:是一个整数数字
数组中的每个数据,称之为数组元素
数组为每个元素进行编号(专业术语叫做索引),该编号从0开始到最大值(数组长度 减 1)
3.数组元素:
数组名称[索引编号]
举例:
array[3]: 数组array中索引编号为3的元素
4.获取数组长度:
数组内部有个length属性,专门记录数组的长度
数组名称.length: 是一个int数字,代表数组的长度

public class Demo06ArrayUse {
public static void main(String[] args) {
/*int num = 10;
System.out.println(num);//10*/

//创建int数组array,元素分别为10,20,30
int[] array = {10,20,30};//10的索引编号是0,20的索引编号是1,30的索引编号是2

System.out.println(array);//[I@1540e19d: 数组在内存空间的地址值,是一个十六进制的整数数字

//打印10
System.out.println(array[0]);//打印数组array中索引编号为0的元素值: 10

//打印20
System.out.println(array[1]);//打印数组array中索引编号为1的元素值: 20

//打印30
System.out.println(array[2]);//打印数组array中索引编号为2的元素值: 30

System.out.println("--------------");

//把10 修改成 100
array[0] = 100;//把整数数字100,存储到数组array中索引编号为0的元素中

//把20 修改成 200
array[1] = 200;//把整数数字200,存储到数组array中索引编号为1的元素中

//把30 修改成 300
array[2] = 300;//把整数数字200,存储到数组array中索引编号为2的元素中


//打印100
System.out.println(array[0]);//打印数组array中索引编号为0的元素值: 100

//打印200
System.out.println(array[1]);//打印数组array中索引编号为1的元素值: 200

//打印300
System.out.println(array[2]);//打印数组array中索引编号为2的元素值: 300

//获取数组的长度
int count = array.length;
System.out.println("数组长度: "+count);
System.out.println("数组长度: "+array.length);

System.out.println(array[array.length-1]);//30
}
}

注意点:

3.数组元素的默认初始化值,new出来在堆中的都有默认值

  • 数组元素是整形:0
  • 数组元素是浮点型:0.0
  • 数组元素是char型:0
1
2
(注意不是'0',而是数0,ASCII码或'\u0000',表现为空)
if(char[0] == 0) true
  • 数组元素是boolean型,false
  • 数组元素是引用类型,null

4.索引:java中的索引基本上都是从0开始的

除了mysql的column是从1开始的。

5.数组是引用类型,数组型数据是对象[Object],数组中的每个对象相当于该对象的成员变量。(数组一经分配空间,元素按照成员变量同样的方式被隐式初始化)

6.int[] a = new int[0];a[0] = 10;// 运行报错,a没有长度,越界异常

7.数组简写形式不能作为实参,int[] a = {1,2};只能一次性定义时使用

1
2
3
4
把{12}当成形参就是int a; a = {1,2}; // 这样不行的
如果数组初始化中不使用运算符new。需要注意:下列写法是错误的。
int[] arr;
arr={1,2,3,4,5};

8.数组的最大索引值为(数组名.length-1)

数组应用场景

  • 如果需要存储大量的数据,例如如果需要读取100个数,那么就需要定义100个变量,显然重复写100次代码,是没有太大意义的。这时候就可以用数组。
  • 当元素较多时为了方便操作这些数组,会先进行来临时存储,所使用的容器就是数组。

数组原理内存图

java中的内存分配

1
2
3
4
5
6
7
1.方法区: 存储可以运行的class文件。
静态区(优先于对象创建,随类的加载而加载静态内容)
2.方法栈(stack): 方法运行时使用的内存,比如main方法运行,进入方法栈中执行。
3.堆内存(heap): 存储对象或者数组,new来创建的,都存储在堆内存。
常量池(String双引号的常量和new的常量位置不一样)
4.寄存器: 给CPU使用
5.本地方法栈: JVM在使用操作系统功能的时候使用,

内存结构简化图:

image-20230922113005849

图解:

image-20200515121702080

一个数组内存图

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
/*
一个数组内存图

注意:
1.数组名称保存数组在堆内存中的地址值
2.通过数组名称找到堆内存中的具体数组,然后通过索引编号找到对应的具体的某个元素

数组元素有默认值:
1.整数: 0
2.小数: 0.0
3.字符: 空白字符
4.布尔: false
*/
public class Demo01OneArray {
public static void main(String[] args) {
//创建int数组one,长度为2
int[] one = new int[2];//第一个元素索引是0,第二个元素索引1

System.out.println(one);//地址值:[I@1540e19d
System.out.println(one[0]);//打印数组one中索引编号为0的元素的值,默认值: 0
System.out.println(one[1]);//打印数组one中索引编号为1的元素的值,默认值: 1

one[0] = 10;//把整数数字10,存储到数组one中索引编号为0的元素中,默认值0被覆盖
one[1] = 20;//把整数数字20,存储到数组one中索引编号为1的元素中,默认值0被覆盖

System.out.println(one);//地址值:[I@1540e19d
System.out.println(one[0]);//打印数组one中索引编号为0的元素的值: 10
System.out.println(one[1]);//打印数组one中索引编号为1的元素的值: 20
}
}

图解:

image-20200515121737000

两个数组内存图

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
/*
两个数组内存图(每个数组都有自己独立的内存空间,互不影响,互不干扰)

注意:
1.数组名称保存数组在堆内存中的地址值
2.通过数组名称找到堆内存中的具体数组,然后通过索引编号找到对应的具体的某个元素

有两套房子,一套在延庆(钥匙是one),一套在平谷(钥匙是two)
one: 你自己拿着 打开的是延庆的房子
two: 你对象拿着 打开的是平谷的房子
*/
public class Demo02TwoArray {
public static void main(String[] args) {
//创建int数组one,长度为2
int[] one = new int[2];//在北京延庆购买了一套两居室的房子,one是打开这套房子的钥匙

System.out.println(one);//[I@1540e19d
System.out.println(one[0]);//0
System.out.println(one[1]);//0

one[0] = 10;
one[1] = 20;

System.out.println(one);//[I@1540e19d
System.out.println(one[0]);//10
System.out.println(one[1]);//20
System.out.println("--------------");

//又创建一个int数组two,长度为2
int[] two = new int[2];//在北京平谷购买了一套两居室的房子,two是打开这套房子的钥匙

System.out.println(two);//[I@677327b6
System.out.println(two[0]);//0
System.out.println(two[1]);//0

two[0] = 100;
two[1] = 200;

System.out.println(two);//[I@677327b6
System.out.println(two[0]);//100
System.out.println(two[1]);//200
}
}

图解:省略,就是把一个对象的内存图复制两份,每个数组之间没有任何关系

两个变量指向一个数组

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
/*
两个引用一个数组内存图

注意1.:
1.数组名称保存数组在堆内存中的地址值
2.通过数组名称找到堆内存中的具体数组,然后通过索引编号找到对应的具体的某个元素

注意2:
1.数组名称保存数组在堆内存空间的地址值
2.使用数组名进行赋值时,传递的是地址值
3.使用数组名作为方法参数和返回值,传递的都是地址值 --------后面讲解

有一套房子,在延庆,但是该房子有两把钥匙,分别是one和two
one: 你自己拿着 打开的是延庆的房子
two: 你对象拿着 打开的是延庆的房子

*/
public class Demo03SameArray {
public static void main(String[] args) {
//创建int数组one,长度为2
int[] one = new int[2];//在北京延庆购买了一套两居室的房子,one是打开这套房子的钥匙

System.out.println(one);//[I@1540e19d
System.out.println(one[0]);//0
one[0] = 10;
System.out.println(one);//[I@1540e19d
System.out.println(one[0]);//10
System.out.println("--------------");

/*
数组变量one保存的是数组在内存空间的地址值,
赋值给新的数组变量two,导致数组变量one和two保存相同的内存地址,
操作的是同一个数组

one相当于是延庆两居室的房子的钥匙,通过钥匙one又配了一把钥匙two
这样:
钥匙one和two,打开的都是延庆两居室的房子
钥匙one: 你自己拿着
钥匙two: 你对象拿着
*/
int[] two = one;

System.out.println(two);//[I@677327b6
System.out.println(two[0]);//10
two[0] = 100;

System.out.println(two);//[I@677327b6
System.out.println(two[0]);//100
System.out.println(one[0]);//100
}
}

图解:

image-20200515121914077

数组操作的常见问题

数组越界异常

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
/*
数组操作的常见问题一:
数组索引越界(超出了范围)异常
1.问题描述: java.lang.ArrayIndexOutOfBoundsException类,数组索引越界异常类
2.产生原因:
使用索引编号范围数组元素时,给出的索引编号不存在(超出了范围)
索引编号范围: 最小值是0,最大值是数组长度 减 1 (one.length - 1)
3.解决方案:
根据控制台的打印信息,找到出现问题的索引,进行修改
*/
public class Demo01ArrayProblem {
public static void main(String[] args) {
//创建int数组array,并初始化
//int[] one = new int[]{100,200,300};
int[] one = {100,200,300};//100的索引编号是0,200的索引编号是1,300的索引编号是2
System.out.println(one);//数组名称one代表数组的内存地址值:[I@1540e19d
System.out.println(one[0]);//100
System.out.println(one[1]);//200
System.out.println(one[2]);//300
//System.out.println(one[5]);//索引5: 不存在,报出数组索引越界异常
}
}

图解:

image-20200515151208671

数组空指针异常

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
/*
数组操作的常见问题二:
空指针异常
1.问题描述: java.lang.NullPointerException类,空指针异常类
2.产生原因:
null是一个引用类型的常量,可以给任意类型引用变量赋值,
当把null赋值给数组变量one之后,数组变量one将不再指向堆内存空间的任何数组,
也就不可以通过one再访问数组元素,只要访问,报出空指针异常

3.解决方案:
(1)不要通过值为null的数组变量,访问数组元素
(2)根据控制台打印的相关异常信息,找到数组变量是null的地方进行修改,
让数组变量指向一个堆内存空间的数组,就可以访问数组元素了
*/
public class Demo02ArrayProblem {
public static void main(String[] args) {
//创建int数组array,并初始化
//int[] one = new int[]{100,200,300};
int[] one = {100,200,300};//100的索引编号是0,200的索引编号是1,300的索引编号是2
System.out.println(one);//数组名称one代表数组的内存地址值:[I@1540e19d
System.out.println(one[0]);//100
System.out.println(one[1]);//200
System.out.println(one[2]);//300

one = null;
System.out.println(one);//null
//System.out.println(one[0]);//错误: 空指针异常
}
}


图解:

image-20200515151340388

数组练习

数组遍历

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
数组遍历:
就是将数组中的每个元素分别获取出来,就是遍历。遍历也是数组操作中的基石。
public class Demo01EachArray {
public static void main(String[] args) {
//创建int数组array,并初始化
//10的索引编号是0,20的索引编号是1,30的索引编号是2,50的索引编号是3
int[] array = {10,20,30,50/*,60,70,80,90,100*/};
System.out.println(array[0]);
System.out.println(array[1]);
System.out.println(array[2]);
System.out.println(array[3]);
System.out.println("-----------");
//发现: 以上代码重复,只有索引0到3是不同的
//可以使用for循环打印0到3的数字
for (int i = 0; i < 4; i++) {
System.out.println("索引编号: "+i+", 对应的元素值: "+array[i]);
}
System.out.println("-----------");
//发现: 以上for循环中数字4写死了,可以使用数组的长度代替
for (int i = 0; i < array.length; i++) {
//System.out.println("索引编号: "+i+", 对应的元素值: "+array[i]);
System.out.println(array[i]);
}

}
}

求三个int数字的最大值

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
需求:
求三个int数字的最大值

实现步骤:
1.定义3int变量a,b,c,并分别初始化
2.假设变量a的值是最大的,把a保存到int变量max中
3.如果变量b的值大于max,说明max中的值目前不是最大的,把b的值赋值给max
4.如果变量c的值大于max,说明max中的值目前不是最大的,把c的值赋值给max
5.打印max的值
public class Demo02Max {
public static void main(String[] args) {
//1.定义3个int变量a,b,c,并分别初始化
int a = 100, b = 200,c = 300;

//2.假设变量a的值是最大的,把a保存到int变量max中
int max = a;

//3.如果变量b的值大于max,说明max中的值目前不是最大的,把b的值赋值给max
if (b > max) {
max = b;
}
//4.如果变量c的值大于max,说明max中的值目前不是最大的,把c的值赋值给max
if (c > max) {
max = c;
}
//5.打印max的值
System.out.println("最大值: "+max);
}
}

图解:

image-20200515144417507

数组获取最大值元素

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
需求:
int数组元素最大值

实现步骤:
1.创建int数组array,并初始化
2.假设索引为0的元素是最大的,保存到int变量max中
3.使用for循环依次获取后面的(从索引1开始)每个元素
3.1如果当前元素 大于 max 说明max中的值已经不是最大的了
3.2把当前元素赋值给变量max
4.for循环结束后打印max的值
public class Demo03ArrayMax {
public static void main(String[] args) {
//1.创建int数组array,并初始化
int[] array = {5,15,2000,10000,100,4000};

//2.假设索引为0的元素是最大的,保存到int变量max中
int max = array[0];

//3.使用for循环依次获取后面的(从索引1开始)每个元素
for (int i = 1; i < array.length; i++) {
//3.1如果当前元素 大于 max 说明max中的值已经不是最大的了
if (array[i] > max) {
//3.2把当前元素赋值给变量max
max = array[i];
}

}
//4.for循环结束后打印max的值
System.out.println("数组最大值: "+max);
}
}

图解分析:

image-20200429152819315

执行流程:

image-20200515151009936

数组中的常见算法

  1. 数组元素的赋值(杨辉三角、回形数等)
  2. 求数值型数组中元素的最大值、最小值、平均数、总和等
  3. 数组的复制、反转、查找(线性查找、二分法查找)
  4. 数组元素的排序算法

杨辉三角

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
public class $_01_杨辉三角 {
public static void main(String[] args) {
/**
* 使用二维数组打印一个 10 行杨辉三角。
* 【提示】
* 1. 第一行有 1 个元素, 第 n 行有 n 个元素
* 2. 每一行的第一个元素和最后一个元素都是 1
* 3. 从第三行开始, 对于非第一个元素和最后一个元
* 素的元素。即:
* yanghui[i][j] = yanghui[i-1][j-1] + yanghui[i-1][j];
*/
// 1.声明并初始化二维数组
int[][] yangHui = new int[10][];

// 2.给数组的元素赋值
for (int i = 0; i < 10; i++) {
yangHui[i] = new int[i + 1];
// 2.1 首末元素赋值
yangHui[i][0] = yangHui[i][i] = 1;
// 2.2 非首末元素赋值
//if (i > 1) {
for (int j = 1; j < yangHui[i].length - 1; j++) {
yangHui[i][j] = yangHui[i - 1][j - 1] + yangHui[i - 1][j];
}
//}
}

// 3.遍历二维数组
for (int i = 0; i < yangHui.length; i++) {
for (int j = 0; j < yangHui[i].length; j++) {
System.out.print(yangHui[i][j] + " ");
}
System.out.println();
}
}
}

回形数

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
public class $_01_回形数 {
public static void main(String[] args) {
int n = 7;
int[][] arr = new int[n][n];

int count = 0; // 要显示的数据
int maxX = n - 1; // x轴的最大下标
int maxY = n - 1; // Y轴的最大下标
int minX = 0; // x轴的最小下标
int minY = 0; // Y轴的最小下标
while (minX <= maxX) {
for (int x = minX; x <= maxX; x++) {
arr[minY][x] = ++count;
}
minY++;
for (int y = minY; y <= maxY; y++) {
arr[y][maxX] = ++count;
}
maxX--;
for (int x = maxX; x >= minX; x--) {
arr[maxY][x] = ++count;
}
maxY--;
for (int y = maxY; y >= minY; y--) {
arr[y][minX] = ++count;
}
minX++;
}

for (int i = 0; i < arr.length; i++) {
for (int j = 0; j < arr.length; j++) {
String space = (arr[i][j] + "").length() == 1 ? "0" : "";
System.out.print(space + arr[i][j] + " ");
}
System.out.println();
}
}
}

数组最大值、最小值、平均数、总和

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
public class $_03_数值 {
public static void main(String[] args) {
/**
* 定义一个int型的一维数组,包含10个元素,分别赋一些随机整数,
* 然后求出所有元素的最大值,最小值,和值,平均值,并输出出来。
* 要求:所有随机数都是两位数。
* 提示;
* [0,1) * 90 ?[0,90) + 10 ? [10,100) ?[10,99]
* (int)(Math.random() * 90 + 10)
*/
int[] arr = new int[10];

for (int i = 0; i < 10; i++) {
arr[i] = (int) (Math.random() * 99 + 10);
}
// 最大值
int maxValue = arr[0];
for (int i = 1; i < arr.length; i++) {
if (maxValue < arr[i]) {
maxValue = arr[i];
}
}
System.out.println("最大值为:" + maxValue);
// 最小值
int minValue = arr[0];
for (int i = 1; i < arr.length; i++) {
if (minValue > arr[i]) {
minValue = arr[i];
}
}
System.out.println("最小值为:" + minValue);
// 和值
int sum = 0;
for (int i = 0; i < arr.length; i++) {
sum += arr[i];
}
System.out.println("和值:" + sum);
// 平均值
System.out.println("平均值为:" + (sum / arr.length));

}
}

数组复制

1
2
3
4
5
6
7
8
9
10
11
public class $_04_数组复制 {
public static void main(String[] args) {
String[] arr = new String[]{"JJ","DD","MM","GG","NN"};

// 数组的复制
String[] arr1 = new String[arr.length];
for (int i = 0; i < arr1.length; i++) {
arr1[i] = arr[i];
}
}
}

数组反转

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
public class $_05_数组反转 {
public static void main(String[] args) {
String[] arr = new String[]{"JJ", "DD", "MM", "GG", "NN"};


// 数组的反转(这里不是倒着遍历,交换位置)
// 方式一:
for (int i = 0; i < arr.length / 2; i++) {
String temp = arr[i];
arr[i] = arr[arr.length - 1 - i];
arr[arr.length - 1 - i] = temp;
}
System.out.println(Arrays.toString(arr));
// 方式二:
for (int i = 0, j = arr.length - 1; i < j; i++, j--) {
String temp = arr[i];
arr[i] = arr[j];
arr[j] = temp;
}
System.out.println(Arrays.toString(arr));
}
}

线性查找/搜索

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
public class $_05_数组查找 {
public static void main(String[] args) {
String[] arr = new String[]{"JJ", "DD", "MM", "GG", "NN"};

String search = "BB";
// 线性查找
boolean flag = true;
for (int i = 0; i < arr.length; i++) {
if (search.equals(arr[i])) {
System.out.println("找到了指定的元素,位置位:"+i);
flag = false;
break;
}
}
if (flag) {
System.out.println("没有找到");
}
}
}

二分法查找

比线性查找快;

前提:要查找的数组必须有序

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
public class $_05_数组查找 {
public static void main(String[] args) {
// 二分法查找
int[] arr = new int[]{-99, -54, -2, 0, 2, 33, 43, 256, 999};
boolean isFlag = true;
int number = 43;
int head = 0; // 初始的首索引
int end = arr.length - 1; // 初始的末索引
while (head <= end) {
int middle = (head + end) / 2;
if (number == arr[middle]) {
System.out.println("找到元素,位置位 :" + middle);
isFlag = false;
break;
} else if (arr[middle] > number) {
end = middle - 1;
} else {
head = middle + 1;
}
}
if (isFlag) {
System.out.println("没有找到");
}
}
}

冒泡排序

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
public class BubbleSortDemo {
public static void main(String[] args) {
// 初始化一个待排序的数组
int[] array = {5, 2, 9, 1, 5, 6};

System.out.println("排序前的数组:");
printArray(array);

// 冒泡排序
bubbleSort(array);

System.out.println("排序后的数组:");
printArray(array);
}

// 冒泡排序方法
public static void bubbleSort(int[] array) {
int n = array.length;
// 外层循环控制比较的轮数
for (int i = 0; i < n - 1; i++) {
boolean swapped = false; // 标志位:用于优化,如果一轮中没有交换,说明数组已排序
// 内层循环控制每轮比较次数
for (int j = 0; j < n - 1 - i; j++) {
// 如果前一个元素比后一个大,则交换
if (array[j] > array[j + 1]) {
int temp = array[j];
array[j] = array[j + 1];
array[j + 1] = temp;
swapped = true; // 标记发生了交换
}
}
// 如果一轮比较中没有交换,提前结束排序
if (!swapped) {
break;
}
}
}

// 打印数组的方法
public static void printArray(int[] array) {
for (int num : array) {
System.out.print(num + " ");
}
System.out.println();
}
}

快速排序

用到了递归

快速排序适合大规模无序数组的排序,特别是在需要原地排序的情况下,是一种高效的选择。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
public class Demo22_数组_快速排序 {
public static void main(String[] args) {
// 初始化一个待排序的数组
int[] array = {9, 3, 7, 5, 6, 2, 8};

System.out.println("排序前的数组:");
printArray(array);

// 调用快速排序方法
quickSort(array, 0, array.length - 1);

System.out.println("排序后的数组:");
printArray(array);
}

// 快速排序方法
public static void quickSort(int[] array, int left, int right) {
if (left < right) {
// 获取分区点
int pivotIndex = partition(array, left, right);

// 对左侧分区进行递归排序
quickSort(array, left, pivotIndex - 1);

// 对右侧分区进行递归排序
quickSort(array, pivotIndex + 1, right);
}
}

// 分区方法
public static int partition(int[] array, int left, int right) {
// 选择最右边的元素作为基准值(pivot)
int pivot = array[right];
int i = left - 1; // i 用于追踪比 pivot 小的元素位置

for (int j = left; j < right; j++) {
if (array[j] <= pivot) {
i++;
// 交换 array[i] 和 array[j]
int temp = array[i];
array[i] = array[j];
array[j] = temp;
}
}

// 将 pivot 放置在正确的位置
int temp = array[i + 1];
array[i + 1] = array[right];
array[right] = temp;

return i + 1; // 返回 pivot 的索引
}

// 打印数组的方法
public static void printArray(int[] array) {
for (int num : array) {
System.out.print(num + " ");
}
System.out.println();
}

}

二维数组

二维数组存储的是一堆一维数组

为什么用二维数组?

1. 表示多维数据

二维数组可以直观地表示需要二维组织的数据。例如:

  • 表格数据:例如 Excel 表中的行和列。
  • 棋盘或地图:如井字棋、围棋、路径规划等。

示例

1
2
3
4
5
int[][] matrix = {
{1, 2, 3},
{4, 5, 6},
{7, 8, 9}
};

上述表示一个 3x3 的表格。

2. 简化数据的组织和操作

使用二维数组可以避免嵌套多个一维数组的复杂性。

优势

  • 快速定位:通过行列索引可以快速访问或更新数据。

    1
    int value = matrix[1][2]; // 获取第二行第三列的值
  • 统一存储:将相关数据集中在一起,便于管理。

3. 适用于特定问题场景

二维数组在以下场景中非常常用:

(1) 数值计算与矩阵运算

  • 矩阵相乘、转置、求逆等。
  • 如图像处理中的像素操作,每个像素点可以看作二维数组中的一个元素。

(2) 图形与几何

  • 表示二维网格(如地形、游戏地图)。
  • 存储像素点或网格节点。

(3) 动态规划问题

  • 动态规划常用二维数组存储状态转移表,记录子问题的结果,避免重复计算。

    示例:计算两个字符串的最小编辑距离。

    1
    int[][] dp = new int[m+1][n+1];

(4) 表格操作

  • 存储多维数据集,如数据库查询结果或统计数据。
  • 使用二维数组可以自然地映射到行和列。

4. 高效的内存访问

在大多数编程语言中,二维数组在内存中是连续存储的(以行优先或列优先顺序),相较于嵌套列表或链表,访问效率更高。

5. 可扩展性

  • 二维数组可以进一步扩展为多维数组(如三维、四维数组),以适应更复杂的场景。
  • 例如,三维数组可以用来表示一个动态变化的三维空间(如时间维度的变化)。

是否总需要二维数组?

二维数组非常方便,但也有局限性:

  • 如果每行的列数不同,可以考虑使用二维列表(ArrayList<ArrayList<T>>)
  • 如果需要更灵活的索引和结构,可以选择使用更高级的数据结构(如 Map 或自定义类)。

动态初始化

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
public class Demo20_二维数组初始化 {
public static void main(String[] args) {
固定列数的二维数组(); // 在声明二维数组时,仅指定行数,列数稍后动态分配。

动态指定行列(); // 如果行数和列数都需要动态分配,可以直接用循环初始化。

}

private static void 动态指定行列() {
// 定义一个3x4的二维数组
int rows = 3, cols = 4;
int[][] arr = new int[rows][cols];

// 动态初始化所有元素
for (int i = 0; i < rows; i++) {
for (int j = 0; j < cols; j++) {
arr[i][j] = i + j; // 示例赋值
}
}
}

private static void 固定列数的二维数组() {
// 定义一个二维数组,有3行,列数可以动态分配
int[][] arr = new int[3][];

// 动态为每行分配不同的列数
arr[0] = new int[2]; // 第1行有2列
arr[1] = new int[3]; // 第2行有3列
arr[2] = new int[4]; // 第3行有4列

// 为数组元素赋值
arr[0][0] = 1;
arr[0][1] = 2;
// 可以根据需要继续赋值...
}
}

静态初始化

1
2
3
4
5
6
7
8
9
10
11
数据类型[][] 数组名 = {
{值1, 值2, 值3, ...},
{值4, 值5, 值6, ...},
...
};

int[][] arr = {
{1, 2, 3},
{4, 5, 6},
{7, 8, 9}
};