此资料是根据黑马程序员Java零基础视频教程提供的笔记进行修改而成,仅用于个人学习。
学习视频:https://www.bilibili.com/video/BV17F411T7Ao/?p=63&vd_source=a0fb36c5a191d7be68335738a8b342a2
1. 方法概述
1.1 方法的概念
- 什么是方法?
方法(method)是程序中最小的执行单元。
比如:若main方法里面只有两条打印语句,那么这两条打印语句不可能只执行其中一条,必须一起执行,以一个方法为执行单元。
-
实际开发中,什么时候用到方法?
重复的代码、具有独立功能的代码可以抽取到方法中。
-
实际开发中,方法有什么好处?
可以提高代码的复用性、提高代码的可维护性。
-
注意:
- 方法必须先创建才可以使用,该过程成为方法定义。
- 方法创建后并不是直接可以运行的,需要手动使用后,才执行,该过程成为方法调用。
2. 方法的定义和调用
2.1 无参数方法定义和调用
-
定义格式:
public static void 方法名 ( ) { //方法名采用小驼峰命令格式 // 方法体; }
-
范例:
public static void oneDayLife ( ) { System.out.println("吃饭"); System.out.println("睡觉"); System.out.println("打豆豆"); System.out.println("学习"); }
-
调用格式:
方法名();
-
范例:
oneDayLife();
-
注意:
方法必须先定义,后调用,否则程序将报错。
2.2 方法定义和调用练习
需求:打印女朋友的基本信息
public class MethodDemo2 {
public static void main(String[] args) {
//调用方法
printGFInfo();
}
//定义方法
public static void printGFInfo(){
System.out.println("迪丽热巴");
System.out.println("女神");
System.out.println("永远18岁");
}
}
2.3 方法执行的过程
执行过程:
看到方法就进入方法,执行完毕回到调用处。
示例一:
public class Test4 {
public static void main(String[] args) {
System.out.println("a");
method();
System.out.println("b");
}
public static void method() {
System.out.println("c");
System.out.println("d");
}
}
结果:
示例二:
public class Test5 {
public static void main(String[] args) {
System.out.println("main方法开始");
method1();
System.out.println("main方法结束");
}
private static void method1() {
System.out.println("method1方法开始");
method2();
System.out.println("method1方法结束");
}
private static void method2() {
System.out.println("method2方法开始");
System.out.println("method2方法结束");
}
}
结果:
2.4 无参数方法的练习
-
需求:设计一个方法用于打印两个数中的较大数
-
思路:
- ①定义一个方法,用于打印两个数字中的较大数,例如getMax()
- ②方法中定义两个变量,用于保存两个数字
- ③使用分支语句分两种情况对两个数字的大小关系进行处理
- ④在main()方法中调用定义好的方法
代码:
public class MethodTest {
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);
}
}
}
3. 带参数方法定义和调用
3.1 带参数方法定义和调用
定义格式:
参数:由数据类型和变量名组成 , 数据类型 变量名
参数范例:int a
public static void 方法名 (参数1) {
方法体;
}
public static void 方法名 (参数1, 参数2, 参数3...) {
方法体;
}
范例:
public static void isEvenNumber(int number){
...
}
public static void getMax(int num1, int num2, int num3){
...
}
-
注意
方法定义时,参数中的数据类型与变量名都不能缺少,缺少任意一个程序将报错。
方法定义时,多个参数之间使用逗号( ,)分隔。
-
调用格式:
方法名(参数); 方法名(参数1,参数2,参数3);
-
范例:
isEvenNumber(10); getMax(10,20,30);
- 方法调用时,参数的数量与类型必须与方法定义中的设置相匹配,否则程序将报错 。
3.2 形参和实参
- 形参:形式参数,方法定义中的参数
等同于变量定义格式,例如:int number
- 实参:实际参数,方法调用中的参数
等同于使用变量或常量,例如: 10 number
用实参初始化形参!!! 实参和形参需要一一对应!!!(数量 、 数据类型)
3.3 带参数方法练习一
- 需求:设计一个方法用于打印两个数中的较大数,数据来自于方法参数
- 思路:
- ①定义一个方法,用于打印两个数字中的较大数,例如getMax()
- ②为方法定义两个参数,用于接收两个数字
- ③使用分支语句分两种情况对两个数字的大小关系进行处理
- ④在main()方法中调用定义好的方法(使用常量)
- ⑤在main()方法中调用定义好的方法(使用变量)
- 代码:
public class MethodTest {
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);
}
}
}
3.4 带参数方法练习二
需求:定义一个方法,求长方形的周长,将结果在方法中进行打印。
public class Test1 {
public static void main(String[] args) {
//目标:根据不同的需求,选择定义无参的方法,还是带参数的方法
getLength(5.2,1.3);
}
//1.我要干嘛?求长方形的周长
//2.我干这件事情,需要什么才能完成?长 宽
public static void getLength(double len, double width){
double result = (len + width) * 2;
System.out.println(result);
}
}
3.5 带参数方法练习三
需求:定义一个方法,求圆的面积,将结果在方法中进行打印。
public class Test2 {
public static void main(String[] args) {
//目标:根据不同的需求,选择定义无参的方法,还是带参数的方法
getArea(1.5);
}
//1.我要干嘛?求圆的面积
//2.我干这件事情,需要什么才能完成? 半径的平方 * π
public static void getArea(double radius){
double result = radius * radius * 3.14;
System.out.println(result);
}
}
4. 带返回值方法的定义和调用
4.1 带返回值方法定义和调用
-
定义格式
public static 数据类型 方法名 ( 参数 ) { return 数据 ; }
-
范例
public static boolean isEvenNumber( int number ) { return true ; } public static int getMax( int a, int b ) { return 100 ; }
-
注意:
- 方法定义时return后面的返回值与方法定义上的数据类型要匹配,否则程序将报错。
-
-
调用格式
方法名 ( 参数 ) ; 数据类型 变量名 = 方法名 ( 参数 ) ; //用变量接收方法返回值
-
范例
isEvenNumber ( 5 ) ; boolean flag = isEvenNumber ( 5 );
-
注意:
- 方法的返回值通常会使用变量接收,否则该返回值将无意义。
-
4.2 带返回值方法练习1
-
需求:设计一个方法可以获取两个数的较大值,数据来自于参数
-
思路:
- ①定义一个方法,用于获取两个数字中的较大数
- ②使用分支语句分两种情况对两个数字的大小关系进行处理
- ③根据题设分别设置两种情况下对应的返回结果
- ④在main()方法中调用定义好的方法并使用变量保存
- ⑤在main()方法中调用定义好的方法并直接打印结果
-
代码:
public class MethodTest { public static void main(String[] args) { //1、在main()方法中调用定义好的方法并使用变量保存 int result = getMax(10,20); System.out.println(result); //20 //2、在main()方法中调用定义好的方法并直接打印结果 System.out.println(getMax(10,20)); //20 } //定义一个方法,用于获取两个数字中的较大数 public static int getMax(int a, int b) { //使用分支语句分两种情况对两个数字的大小关系进行处理 //根据题设分别设置两种情况下对应的返回结果 if(a > b) { return a; } else { return b; } } }
4.3 带返回值方法练习2
需求:
定义一个方法,求一家商场每个季度的营业额。根据方法结果再计算出全年营业额。
代码示例:
package com.itheima.demo;
public class MethodDemo9 {
public static void main(String[] args) {
/*需求:定义一个方法,求一家商场每个季度的营业额。
根据方法结果再计算出全年营业额。*/
//第一个季度的营业额
int sum1 = getSum(10, 20, 30);
//第一个季度的营业额
int sum2 = getSum(10, 20, 30);
//第一个季度的营业额
int sum3 = getSum(10, 20, 30);
//第一个季度的营业额
int sum4 = getSum(10, 20, 30);
int sum = sum1 + sum2 + sum3 + sum4;
System.out.println(sum);
}
//心得:
//1.我要干嘛? 决定了方法体 每个季度的营业额
//2.我干这件事情,需要什么才能完成? 决定了形参 需要三个月的营业额 a b c
//3.我干完这件事情,看调用处是否需要使用方法的结果。 决定了返回值
//如果需要使用,那么必须返回
//如果不需要使用,可以返回也可以不返回
public static int getSum(int month1,int month2,int month3){
int sum = month1 + month2 + month3;
//因为方法的调用处,需要继续使用这个结果
//所以我们必须要把sum返回
return sum;
}
}
4.4 带返回值方法练习3
需求:
键盘录入两个圆的半径(整数),比较两个圆的面积。
代码示例:
import java.util.Scanner;
public class MethodDemo10 {
public static void main(String[] args) {
//需求:键盘录入两个圆的半径(整数),比较两个圆的面积。
//键盘录入圆的半径
Scanner sc = new Scanner(System.in);
System.out.println("请输入第一个圆的半径");
int radii1 = sc.nextInt();
System.out.println("请输入第二个圆的半径");
int radii2 = sc.nextInt();
double area1 = getArea(radii1);
double area2 = getArea(radii2);
//比较的处理不放在自定义方法里面,为了后续的可维护性!!!
if(area1 > area2){
System.out.println("第一个圆更大");
}else{
System.out.println("第二个圆更大");
}
}
//心得:
//1.我要干嘛? 求圆的面积
//2.我干这件事情,需要什么才能完成? 半径
//3.方法的调用处,是否需要继续使用方法的结果 要比较
public static double getArea(int radii) {
double area = 3.14 * radii * radii;
return area;
}
}
5. 方法的注意事项
5.1 方法的注意事项
- 方法不调用就不执行
- 方法的编写顺序和执行顺序无关
-
方法与方法之间是平级关系,方法不能嵌套定义
示例代码:
public class MethodDemo { public static void main(String[] args) { } public static void methodOne() { public static void methodTwo() { //不能嵌套!!! // 这里会引发编译错误!!! } } }
-
void表示无返回值,可以省略return,也可以单独的书写return,后面不加数据,表示结束方法。
-
示例代码:
public class MethodDemo { public static void main(String[] args) { } public static void methodTwo() { //return 100; 编译错误,因为吃方法没有具体的返回值类型 return; //System.out.println(100); return语句后面不能跟数据或代码 } }
-
- return语句下面不能编写代码,因为永远执行不到,属于无效的代码。
5.2 方法的通用格式
-
格式:
public static 返回值类型 方法名(参数) { 方法体; return 数据 ; }
-
解释:
名词 含义 public static 修饰符,目前先记住这个格式 返回值类型 方法操作完毕之后返回的数据的数据类型,如果方法操作完毕,没有数据返回,这里写void,而且方法体中一般不写return 方法名 调用方法时候使用的标识 参数 由数据类型和变量名组成,多个参数之间用逗号隔开 方法体 完成功能的代码块 return 如果方法操作完毕,有数据返回,用于把数据返回给调用者 -
定义方法时,要做到两个明确
- 明确返回值类型:主要是明确方法操作完毕之后是否有数据返回,如果没有,写void;如果有,写对应的数据类型
- 明确参数:主要是明确参数的类型和数量
-
调用方法时的注意:
- void类型的方法,直接调用即可
- 非void类型的方法,推荐用变量接收调用
6. 方法重载
6.1 方法重载
-
方法重载概念
方法重载指同一个类中定义的多个方法之间的关系,满足下列条件的多个方法相互构成重载
-
多个方法在同一个类中
-
多个方法具有相同的方法名
-
多个方法的参数不相同:参数类型不同、数量不同、顺序不同
-
-
注意:
- 重载仅对应方法的定义,与方法的调用无关,调用方式参照标准格式
- 重载仅针对同一个类中方法的名称与参数进行识别,与返回值无关,换句话说不能通过返回值来判定两个方法是否相互构成重载
-
正确范例:
public class MethodDemo { /*参数类型不同*/ public static void fn(int a) { //方法体 } public static int fn(double a) { //方法体 } } public class MethodDemo { /*参数个数不同*/ public static float fn(int a) { //方法体 } public static int fn(int a , int b) { //方法体 } } public class MethodDemo { /*参数顺序不同 不建议使用!!!!!!*/ public static int fn(double a,int b) { //方法体 } public static int fn(int a , double b) { //方法体 } }
-
错误范例:
public class MethodDemo { public static void fn(int a) { //方法体 } public static int fn(int a) { /*错误原因:重载与返回值无关*/ //方法体 } } public class MethodDemo01 { public static void fn(int a) { //方法体 } } public class MethodDemo02 { public static int fn(double a) { /*错误原因:这是两个类的两个fn方法*/ //方法体 } }
6.2 方法重载练习
-
需求:使用方法重载的思想,设计比较两个整数是否相同的方法,兼容全整数类型(byte,short,int,long)
-
思路:
- ①定义比较两个数字的是否相同的方法compare()方法,参数选择两个int型参数
- ②定义对应的重载方法,变更对应的参数类型,参数变更为两个long型参数
- ③定义所有的重载方法,两个byte类型与两个short类型参数
- ④完成方法的调用,测试运行结果
-
代码:
public class MethodTest { public static void main(String[] args) { //调用方法 System.out.println(compare(10, 20)); /* 或者使用变量 byte a = 10; byte b = 20; System.out.println(compare(b1,b2)); */ System.out.println(compare((byte) 10, (byte) 20)); //强制转换 System.out.println(compare((short) 10, (short) 20)); System.out.println(compare(10L, 20L)); } //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; } }
7. 方法训练题目
7.1 练习一:数组遍历
-
需求:设计一个方法用于数组遍历,要求遍历的结果是在一行上的。例如:[11, 22, 33, 44, 55]
-
思路:
-
①因为要求结果在一行上输出,所以这里需要在学习一个新的输出语句System.out.print(“内容”);
System.out.println(“内容”); 输出内容并换行
System.out.print(“内容”); 输出内容不换行
System.out.println(); 起到换行的作用
-
②定义一个数组,用静态初始化完成数组元素初始化
-
③定义一个方法,用数组遍历通用格式对数组进行遍历
-
④用新的输出语句修改遍历操作
-
⑤调用遍历方法
-
-
代码:
public class Test1 { public static void main(String[] args) { /* //先打印数据,再进行换行 System.out.println("aaa"); //只打印不换行 System.out.print("bbb"); System.out.print("ddd"); //不打印任何内容,只换行 System.out.println();*/ //设计一个方法用于数组遍历,要求遍历的结果是在一行上的。例如:[11, 22, 33, 44, 55] int[] arr = { 1,2,3,4,5}; printArr(arr); } //1.我要遍历数组 //2.需要什么? 数组 //3.调用处是否需要使用方法的结果。 public static void printArr(int[] arr){ System.out.print("["); //开始括号 for (int i = 0; i < arr.length; i++) { if(i == arr.length - 1){ System.out.println(arr[i] + "]"); //结尾括号 }else{ System.out.print(arr[i] + ", "); } } } }
7.2 练习二:数组最大值
-
需求:设计一个方法用于获取数组中元素的最大值
-
思路:
- ①定义一个数组,用静态初始化完成数组元素初始化
- ②定义一个方法,用来获取数组中的最大值,最值的认知和讲解我们在数组中已经讲解过了
- ③调用获取最大值方法,用变量接收返回结果
- ④把结果输出在控制台
-
代码:
public class MethodTest02 { public static void main(String[] args) { //定义一个数组,用静态初始化完成数组元素初始化 int[] arr = { 12, 45, 98, 73, 60}; //调用获取最大值方法,用变量接收返回结果 int number = getMax(arr); //把结果输出在控制台 System.out.println("最大值number:" + number); } //定义一个方法,用来获取数组中的最大值 /* 两个明确: 返回值类型:int 参数:int[] arr */ public static int getMax(int[] arr) { int max = arr[0]; //先默认第一个元素最大 for(int x=1; x<arr.length; x++) { if(arr[x] > max) { max = arr[x]; } } return max; } }
7.3 练习三:返回索引
需求:
定义一个方法获取数字在数组中的索引位置,将索引结果返回给调用处,如果有重复的,只要获取第一个即可。
代码示例:
package com.itheima.demo;
public class Test4 {
public static void main(String[] args) {
//定义一个方法获取数字,在数组中的索引位置,将结果返回给调用处
//如果有重复的,只要获取第一个即可
int[] arr = {
1,2,3,4,5};
int index = contains(arr, 3); //是否包含数字3
if(index == -1){
System.out.println("数组中不包含该数!");
}
else{
System.out.println("该数的索引是:"+index);
}
}
//1. 我要干嘛?判断数组中的某一个数是否存在
//2. 需要什么?数组 数字
//3. 调用处是否需要继续使用?返回
//获取number在arr中的位置
public static int contains(int[] arr, int number) {
//遍历arr得到每一个元素
for (int i = 0; i < arr.length; i++) {
//拿着每一个元素跟number比较
if(arr[i] == number){
//如果相等,表示找到了,返回索引
return i;
}
}
//当循环结束之后,如果还不能返回索引,表示数组中不存在该数据
//可以返回-1
return -1;
}
}
//return break关键字的区别
return:其实跟循环没有什么关系,跟方法有关的,表示1结束方法 2返回结果
//如果方法执行到了return,那么整个方法全部结束,里面的循环也会随之结束了。
//break关键字
//其实跟方法没有什么关系,结束循环或者switch的。
//如果找到了对应的索引,使用break的话,仅仅是结束循环,方法还没有结束掉!!! 使用return 才可以全部结束掉!!!
7.4 练习四:复制数组
需求:
定义一个方法copyOfRange(int[]arr,int from, int to)
功能:
将数组arr中从索引from(包含from)开始,到索引to结束(不包含to)的元素复制到新数组中,将新数组返回。
public static void main(String[] args) {
//1.定义原数组
int[] array = {
1,2,3,4,5,6,7,8,9};
//2.调用方法拷贝数据
int[] new_array = copyOfRange(array, 3, 8);//注意不要越界访问
//3.遍历数组
for (int i = 0; i < new_array.length; i++) {
System.out.print(new_array[i]+" ");
}
}
public static int[] copyOfRange(int[]array, int from, int to){
//1.定义数组
int[] new_array = new int[to - from];
//伪造索引
int index = 0;
for (int i = from; i < to; i++) {
new_array[index] = array[i];
index++;
}
//返回该数组
return new_array;
}
8. 方法的内存
8.1 方法调用的基本内存原理
8.1.1 进栈:先进后出
如:弹夹压子弹
8.1.2 出栈:后进先出
8.1.3 程序运行时内存变化
首先main方法进栈,调用eat方法,eat方法进栈,调用study方法,study方法进栈,study方法打印“学习”,执行完后,study出栈,回到调用处,eat方法继续打印“吃饭”,继续调用sleep方法,sleep进栈,sleep方法打印“睡觉”,执行完后,sleep出栈,回到调用处,此时eat方法执行完毕,eat方法出栈,回到调用处,最后main方法执行完,main方法出栈。
8.2 方法传递基本数据类型的内存原理
8.2.1 基本数据类型
- 整数类型
- 浮点数类型
- 布尔类型
- 字符类型
变量中存储的是真实的数据,数据值是存储在自己的空间中。
特点:赋值给其他变量,也是赋的真实的值。
8.2.2 引用数据类型
引用:使用了其他空间中的数据,除了上面讲到的4种基本数据类型。
- new出来的都是引用数据类型。
- 数据值是存储在其他空间中,自己空间中存储的是地址值。(变量中存储的是地址值)
特点:赋值给其他变量,赋的地址值。
8.2.3 从内存的角度区分
8.2.4 参数的传递
传递基本数据类型时,传递的是真实的数据,形参的改变,不影响实际参数的值。
示例:
public static void main(String[] args) {
int number = 100;
System.out.println("调用change方法前:"+number); //100
change(number);
System.out.println("调用change方法前:"+number); //100
}
private static void change(int number) {
number = 200; //只是改变了形参的数据,实参数据不会不改变
}
想要改变:
public static void main(String[] args) {
int number = 100;
System.out.println("调用change方法前:"+number); //100
number = change(number); //接收返回的数据,相当于重新给number赋值了
System.out.println("调用change方法前:"+number); //200
}
private static int change(int number) {
number = 200; //只是改变了形参的数据,实参数据不会不改变
return number; //但是返回了这个数据
}
8.3 方法传递引用数据类型的内存原理
传递引用数据类型时,传递的是地址值,形参的改变,影响实际参数的值。
示例:
public static void main(String[] args) {
int[] arr = {
10,20,30};
System.out.println("调用change方法前:"+arr[1]); //20
change(arr);
System.out.println("调用change方法后:"+arr[1]); //200
}
private static void change(int[] arr) {
//指向的同一个地址空间
arr[1] = 200; //传递引用数据类型时,传递的是地址值,形参的改变,影响实际参数的值
}