tips: 下一篇基础知识请点击链接
数组
什么是数组
数组就是存储数据长度固定的容器,存储多个数据的数据类型要一致。
数组定义格式
第一种
数据类型[ ] 数据命名
int[] arr;
double[] arr;
char[] arr;
第二种
数据类型 数据名[ ]
int arr[];
double arr[];
char arr[];
数组动态初始化
什么是动态初始化
数组动态初始化就是只给定数组的长度,由系统给出默认初始化值。
动态初始化格式
数据类型[] 数组名 = new 数据类型[数组长度]
int[] arr = new int[6];
动态初始化格式详解
- 等号左边:
- int:数组的数据类型
- [ ]:代表这是一个数组
- arr:代表数组的名称
- 等号右边:
- new:为数组开辟内存空间
- int:数组的数据类型
- [ ]:代表这是一个数组
- 6:代表数组的长度
数组动态初始化
什么是索引
每一个存储到数组的元素,都会自动的拥有一个编号,从0开始。这个自动编号称为数组索引(index),可以通过数组的索引访问到数组中的元素。
访问数组元素格式
数组名[索引];
int[] arr = {111,222,333}
0 => 111
1 => 222
2 => 333
示例代码
public class ArrayDemo {
public static void main (String[] args) {
int[] arr = new int[2];
//输出数组名
System.out.println(arr); // 对象地址
//输出数组中的元素
System.out.println(arr[0]);
System.out.println(arr[1]);
}
}
内存分配
内存概述
内存是计算机中的重要原件,临时存储区域,作用是运行程序。我们编写的程序是存放在硬盘中的,在硬盘中的程序是不会运行的。必须放进内存中才能运行,运行完毕后会清空内存。Java虚拟机要运行程序,必须要对内存进行空间的分配和管理。
java中的内存分配
JVM会将内存划分成5个区域。目前只需要记住两个内存,分别是:栈内存和堆内存
区域名称 | 作用 |
---|---|
寄存器 | 给CPU使用,和开发无关 |
本地方法栈 | JVM在使用操作系统功能的时候使用,和开发无关 |
方法区 | 存储可以运行的class文件 |
堆内存 | 存储对象或者数组,new来创建的,都存储在堆内存 |
方法栈 | 方法运行时使用的内存,比如main方法运行,进入方法栈中执行 |
单个数组的内存图
new出来的数组保存在堆中,通过地址找到这个数组,通过数组的索引可以操作数组中的元素
public class Demo {
public static void main (String[] args) {
int[] arr = new int[2]
System.out.println(arr); // [I@b684288
/* [:数组
I:int类型数组
@:没有含义,起分割作用
b684288:数组的内存地址
*/
System.out.println(arr[0]); // 0
System.out.println(arr[1]); // 0
arr[1] = 8;
System.out.println(arr[1]); // 8
}
}
方法栈(栈) 堆内存(堆)
main方法
int[] arr = 0x666 ===============> 0x666 => new int[2]
System.out.println(arr[0]); // 0 0 => 0
System.out.println(arr[1]); // 0 1 => 0
arr[1] = 8;
System.out.println(arr[1]); // 8 1 => 8
多个数组的内存图
- new出来的数组存储在堆中
- new一次新建一个数组
- new两次新建两个数组,这两个数组互不干扰
多个数组指向相同内存图
- int[ ] arr2 = arr1
- 没有new就没有创建创建新数组
- =是赋值运算,把右边的值赋值给左边的变量
- 把arr1的值赋值给arr2,所以arr1和arr2保存相同的值
数组静态初始化
什么是静态初始化
在创建数组时,直接将元素确定
静态初始化格式
- 完整版格式
数据类型[] 数组名 = new 数据类型[]{元素1,元素2,...};
- 简化版格式
数据类型[] 数组名 = {元素1,元素2,...};
数组操作的两个常见小问题
索引越界异常
- 出现原因
public class Demo {
public static void main(String[] args) {
int[] arr = new int[3];
System.out.println(arr[3]);
}
}
数组长度为3,索引范围是0-2,但是却访问了一个3的索引。程序运行后,将会抛出ArrayIndexOutOfBoundsException数组越界异常。在开发中,数组的越界异常是不能出现的,一旦出现了,就必须要修改编写的代码。
- 解决方案
将错误的索引修改为正确的索引范围即可。
空指针异常
- 出现原因
public class ArrayDemo {
public static void main(String[] args) {
int[] arr = new int[3];
//把null赋值给数组
arr = null;
System.out.println(arr[0]);
}
}
arr = null这行代码,意味着变量arr将不会再保存数组的内存地址,也就不允许再操作数组了,因此运行的时候会抛出NullPoin terException空指针异常。在开发中,数组的空指针异常是不能出现的,一旦出现,就必须要修改编写的代码。
- 解决方案
给数组一个真正的堆内存空间引用即可。
数组遍历
- 数组遍历:就是将数组中的每个元素分别获取出来,就是遍历。遍历也是数组操作中的基石。
public class Demo {
public static void main(String[] args) {
int[] arr = { 1, 2, 3 };
System.out.println(arr[0]);
System.out.println(arr[1]);
System.out.println(arr[2]);
}
}
以上代码是可以将数组中每个元素全部遍历出来,但是如果数组元素非常多,这种写法肯定不行,因此需要改造成循环的写法。数组的索引是 0 到 lenght-1 ,可以作为循环的条件出现。
public class Demo {
public static void main(String[] args) {
//定义数组
int[] arr = { 1, 2, 3 };
//使用通用的遍历格式
for(int x = 0; x < arr.length; x++) {
System.out.println(arr[x]);
}
}
}
- 数组的最大索引:
最大索引 = length - 1 - IDEA操作
遍历数组:数组名.fori
快捷生成for循环:次数.fori
数组最值
- 最大值获取:从数组的所有元素中找出最大值。
- 实现思路:
- 定义变量,保存数组0索引上的元素
- 遍历数组,获取出数组中的每个元素
- 将遍历到的元素和保存数组0索引上值的变量进行比较
- 如果数组元素的值大于变量的值,变量记录新的值
- 数组循环遍历结束,变量保存的就是数组中的最大值
- 代码实现:
public class Demo {
public static void main(String[] args) {
//定义数组
int[] arr = {10, 33, 101, 68, 47};
//定义一个变量,用于保存最大值
//取数组中第一个数据作为变量的初始值
int max = arr[0];
//与数组中剩余的数据逐个比对,每次比对将最大值保存到变量中
for(int x=1; x<arr.length; x++) {
if(arr[x] > max) {
max = arr[x];
}
}
//循环结束后打印变量的值
System.out.println("max:" + max);
}
}
方法
方法的概念
- 方法(method)是将具有独立功能的代码块组织成为一个整体,使其具有特殊功能的代码集,一般来说一个功能就做成一个方法,可以重复使用,提高代码的复用性
- 注意:
- 方法必须先创建才可以使用,该过程成为方法定义
- 方法创建后并不是直接可以运行的,需要手动使用后,才执行,该过程成为方法调用
方法的定义和调用
无参数方法定义和调用
- 定义格式:
public static void 方法名 () {
// 方法体;
}
- 范例:
public static void method () {
// 方法体;
}
- 调用格式:
方法名();
- 范例:
method();
- 注意:
方法必须先定义,后调用,否则程序将报错
方法调用过程
每个方法在被调用执行的时候,都会进入栈内存,并且拥有自己独立的内存空间,方法内部代码调用完毕之后,会从栈内存中弹栈消失。
无参数方法的练习
- 需求L:设计一个方法用于打印两个数中的较大数
- 思路:
- 定义一个方法,用于打印两个数字中的较大数,例如getMax()
- 方法中定义两个变量,用于保存两个数字
- 使用分支语句分两种情况对两个数字的大小关系进行处理
- 在main()方法中调用定义好的方法
- 代码示例:
public class Demo {
public static void main(String[] args) {
//在main()方法中调用定义好的方法
getMax();
}
//定义一个方法,用于打印两个数字中的较大数,例如getMax()
public static void getMax() {
//方法中定义两个变量,用于保存两个数字
int a = 10;
int b = 20;
//使用分支语句分两种情况对两个数字的大小关系进行处理
if(a > b) {
System.out.println(a);
} else {
System.out.println(b);
}
}
}
带参数方法定义和调用
带参数方法定义和调用
- 定义格式:
参数:由数据类型和变量名组成 => 数据类型 变量名
参数范例:int a
public static void 方法名 (参数1) {
方法体;
}
public static void 方法名 (参数1, 参数2, 参数3,...) {
方法体;
}
- 范例:
public static void isNumber(int number){
...
}
public static void getMax(int num1, int num2){
...
}
- 注意:
- 方法定义时,参数中的数据类型与变量名都不能缺少,缺少任意一个程序将报错
- 方法定义时,多个参数之间使用逗号( , )分割
- 调用格式:
方法名(参数);
方法名(参数1,参数2);
-范例:
isNumber(100);
getMax(100,200);
方法在调用时,参数的数量与类型必须与方法定义中的设置相匹配,否则程序将报错
形参和实参
- 形参:方法定义中的参数
等同于变量定义格式,例如:int number - 实参:方法调用中的参数
等同于使用变量或常量。例如:100 number
带参数方法练习
- 需求:设计一个方法用于打印两个数中的较大数,数据来自于方法参数
- 思路:
- 定义一个方法,用于打印两个数字中的较大数,例如getMax()
- 为方法定义两个参数,用于接收两个数字
- 使用分支语句分两种情况对两个数字的大小关系进行处理
- 在main()方法中调用定义好的方法(使用常量)
- 在main()方法中调用定义好的方法(使用变量)
- 示例代码:
public class Demo {
public static void main(String[] args) {
//在main()方法中调用定义好的方法(使用常量)
getMax(10,20);
//调用方法的时候,要几个,就给几个,要什么类型的,就给什么类型的
//getMax(30);
//getMax(10.0,20.0);
//在main()方法中调用定义好的方法(使用变量)
int a = 10;
int b = 20;
getMax(a, b);
}
//定义一个方法,用于打印两个数字中的较大数,例如getMax()
//为方法定义两个参数,用于接收两个数字
public static void getMax(int a, int b) {
//使用分支语句分两种情况对两个数字的大小关系进行处理
if(a > b) {
System.out.println(a);
} else {
System.out.println(b);
}
}
}
带返回值方法的定义和调用
带返回值方法定义和调用
- 定义格式:
public static 数据类型 方法名 ( 参数 ) {
return 数据 ;
}
- 范例:
public static boolean isNumber( int number ) {
return true ;
}
public static int getMax( int a, int b ) {
return 100 ;
}
注意:方法定义时return后面的返回值与方法定义上的数据类型要匹配,否则程序将报错
- 调用格式
方法名 ( 参数 ) ;
数据类型 变量名 = 方法名 ( 参数 ) ;
- 范例:
isNumber ( 10 ) ;
boolean flag = isNumber ( 10 );
注意:方法的返回值通常会使用变量接收,否则该返回值将无意义
带返回值方法练习
- 需求:设计一个方法可以获取两个数的较大值,数据来自于参数
- 思路:
- 定义一个方法,用于打印两个数字中的较大数
- 使用分支语句分两种情况对两个数字的大小关系进行处理
- 根据题设分别设置两种情况下对应的返回结果
- 在main()方法中调用定义好的方法并使用变量保存
- 在main()方法中调用定义好的方法并直接打印结果
- 示例代码:
public class Demo {
public static void main(String[] args) {
//在main()方法中调用定义好的方法并使用变量保存
int result = getMax(100,200);
System.out.println(result);
//在main()方法中调用定义好的方法并直接打印结果
System.out.println(getMax(100,200));
}
//定义一个方法,用于获取两个数字中的较大数
public static int getMax(int a, int b) {
//使用分支语句分两种情况对两个数字的大小关系进行处理
//根据题设分别设置两种情况下对应的返回结果
if(a > b) {
return a;
} else {
return b;
}
}
}
方法的注意事项
注意事项
- 方法不能嵌套定义
public class Demo {
public static void main(String[] args) {
}
public static void methodOne() {
public static void methodTwo() {
// 这里会引发编译错误!!!
}
}
}
- void表示无返回值,可以省略return,也可以单独的书写return,后面不加数据
public class Demo {
public static void main(String[] args) {
}
public static void methodTwo() {
//return 100; 编译错误,因为没有具体返回值类型
return;
//System.out.println(100); return语句后面不能跟数据或代码
}
}
通用格式
- 格式:
public static 返回值类型 方法名( 参数 ) {
方法体;
return 数据 ;
}
- 解释:
- public static 修饰符
- 返回值类型 方法操作完毕后返回的数据的数据类型,如果方法操作完毕,没有数据返回,这里写void,而且方法体中一般不写return
- 方法名 调用方法时候使用的标识
- 参数 由数据类型和变量名组成,多个参数之间用逗号隔开
- 方法体 完成功能的代码块
- return 如果方法操作完毕,有数据返回,用于把数据返回给调用者
- 定义方法时,要做到两个明确
- 明确返回值类型:主要是明确方法操作完毕之后是否有数据返回,如果没有,写void;如果有,写对应的数据类型
- 明确参数:主要是明确参数的类型和数量
- 调用方法时的注意:
- void类型的方法,直接调用即可
- 非void类型的方法,推荐用变量接收调用
方法重载
方法重载概念
- 概念
方法重载指同一个类中定义的多个方法之间的关系,满足下列条件的多个方法相互构成重载 - 多个方法在同一个类中
- 多个方法具有相同的方法名
- 多个方法的参数顺序不同,类型不同或者数量不同
- 注意
- 重载仅对应方法的定义,与方法的调用无关,调用方式参照标准格式
- 重载仅针对同一个类中方法的名称与参数进行识别,与返回值无关,换句话说不能通过返回值来判定两个方法是否相互构成重载
- 正常范例
public class Demo {
public static void method(int a) {
//方法体
}
public static int method(double a) {
//方法体
}
}
public class Demo {
public static float method(int a) {
//方法体
}
public static int method(int a, int b) {
//方法体
}
}
- 错误范例
public class Demo {
public static void method(int a) {
//方法体
}
public static int method(int a) { /*错误原因:重载与返回值无关*/
//方法体
}
}
public class Demo01 {
public static void method(int a) {
//方法体
}
}
public class Demo02 {
public static int method(double a) { /*错误原因:这是两个类的两个fn方法*/
//方法体
}
}
方法重载练习
- 需求:使用方法重载的思想,设计比较两个整数是否相同的方法,兼容全整数类型(byte,short,int,long
- 思路:
- 定义比较两个数字的是否相同的方法compare()方法,参数选择两个int型参数
- 定义对应的重载方法,变更对应的参数类型,参数变更为两个long型参数
- 定义所有的重载方法,两个byte类型与两个short类型参数
- 完成方法的调用,测试运行结果
- 代码:
public class Demo {
public static void main(String[] args) {
//调用方法
System.out.println(compare(100, 200));
System.out.println(compare((byte) 100, (byte) 200));
System.out.println(compare((short) 100, (short) 200));
System.out.println(compare(100L, 200L));
}
//int
public static boolean compare(int a, int b) {
System.out.println("int");
return a == b;
}
//byte
public static boolean compare(byte a, byte b) {
System.out.println("byte");
return a == b;
}
//short
public static boolean compare(short a, short b) {
System.out.println("short");
return a == b;
}
//long
public static boolean compare(long a, long b) {
System.out.println("long");
return a == b;
}
}
方法的参数传递
方法参数传递基本类型
- 测试代码:
public class Demo {
public static void main(String[] args) {
int num = 100;
System.out.println("调用change方法前:" + num);
change(num);
System.out.println("调用change方法后:" + num);
}
public static void change(int num) {
num = 200;
}
}
- 结论:
基本数据类型的参数,形式参数的改变,不影响实际参数 - 依据:
每个方法在栈内存中,都会有独立的栈空间,方法运行结束后就会弹栈消失
方法参数传递引用类型
- 测试代码:
public class Demo {
public static void main(String[] args) {
int[] arr = {100, 200, 300};
System.out.println("调用change方法前:" + arr[1]);
change(arr);
System.out.println("调用change方法后:" + arr[1]);
}
public static void change(int[] arr) {
arr[1] = 2000;
}
}
- 结论:
对于引用类型的参数,形式参数的改变,影响实际参数的值 - 依据:
引用数据类型的传参,传入的是地址值,内存中会造成两个引用指向同一个内存的效果,所以即使方法弹栈,堆内存中的数据也已经是改变后的结果