1. 什么是方法
在前面我们已经了解过方法的概念,Java 程序的入口main()
就是一个方法。System.out.println();
语句中println()
也是一个方法。
如果你使用IntelliJ IDEA
查看代码,可以使用Ctrl
+ 鼠标左键单击代码中的 println()
方法,查看 JDK 源码中对于这个方法的定义:
上面的截图就是我们经常调用的老朋友println()
方法的代码实现,我们可以将方法理解为一个命名语句块,通过其名称 + 括号运算符()
可以调用。我们可以将需要重复编写的代码,封装到一个方法中。提高代码的复用性。
2. 如何定义方法
在 Java 中,定义一个方法的语法为:
- 1
- 2
- 3
- 4
访问修饰符有 4 种情况:
public
、private
、protected
,也可以省略(default
)。由于涉及到后面的面向对象相关知识,本节统一使用public
修饰方法;返回类型:可以是任何的数据类型或
void
,如果方法没有返回值,返回类型设置为void
;方法名:方法名的命名规范和变量相同;
参数列表:参数是变量的一种类型,参数变量的作用域在方法内部;
方法体:方法内部的一些语句。当方法返回值为
void
时,可以省略return
语句。
3. 分类
根据方法是否带有参数、是否有返回值,可以分为 4 类:
- 无参无返回值方法
- 无参带返回值方法
- 带参数无返回值方法
- 带参数带返回值方法
3.1 无参无返回值方法
无参无返回值方法,即参数列表为空,返回值为void
的方法。例如,我们经常需要输出一段内容。可以将输出语句封装到方法中:
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
上面代码,在类MethodDemo1
中定义了一个方法printText
,其中有一些打印语句。那么如何调用这个方法,让其执行呢?
对于非静态方法(即非static
关键字修饰的方法),首先要使用new
关键字实例化类,生成一个对象,再通过对象名.方法名()
的方式去调动方法。
如下是 MethodDemo1 类方法调用的实例:
运行结果:
- 1
- 2
- 3
运行结果:
- 1
- 2
- 3
3.2 无参带返回值方法
无参带返回值方法,即参数列表为空,返回值不为void
的方法,方法体内部要使用return
语句返回其声明的返回类型。
我们来自定义一个返回Hello World!
字符串的方法:
运行结果:
- 1
3.3 带参数无返回值方法
带参数无返回值方法,即带有参数列表,返回值为void
的方法。
参数列表可能有一个或多个参数,多个参数使用逗号(,
)隔开。定义方式为:DataType 参数名1, DataType 参数名2
,DateType 为参数的类型。
下面是一个根据长和宽求长方形面积的方法实例:
运行结果:
- 1
- 2
在调用带参数方法时,有两种传参方式:
- 先初始化变量,再将变量传递给方法:
- 1
- 2
- 3
- 直接传入对应类型字面量:
- 1
当调用方把参数传递给方法时,调用时传递的值会按参数位置一一绑定,这也比较符合我们正常的思维逻辑。因此,对于有多个参数的方法,一定要注意每个参数的位置。
3.4 带参数带返回值方法
带参带返回值方法,即带有参数列表,返回值不为void
的方法。
下面我们来看一个实例,求从1到n的平方和:
运行结果:
- 1
我们也可以不定义变量接收返回结果,直接打印方法调用语句:
- 1
上面有参方法的例子,参数都是简单的基本数据类型,当参数为数组时,我们再来看一个实例:
运行结果:
- 1
代码解析:
上述代码定义了一个查找数组中是否包含指定元素的方法,它将数组和指定元素作为方法参数,并且返回了一个布尔型的结果,方法体内部首先声明了一个待返回的布尔变量,默认为false
,然后对数组参数进行迭代,只要找到与元素参数相同的元素,即更新返回变量为true
,跳出循环;如果数组迭代完成后依旧未找到指定的元素,那么待返回变量依然是false
,最后返回这个变量。在方法调用时,以一个布尔变量来接收结果,后续代码根据这个布尔变量做了逻辑判断,以执行不同的语句块。
4. 方法传值
调用方和方法之间有参数的传递时,要注意方法传值问题。
4.1 基本类型的传值
基本类型参数的传递,是调用方值的复制。双方各自的后续修改,互不影响。简单来讲,方法内部对参数变量的任何操作,都不会影响到方法外部的变量。
我们看一个实例:
运行结果:
- 1
- 2
- 3
我们根据输出的结果验证了以上给出的结论。整型参数speed
是一个基本数据类型,在speedUp
方法内部进行了自增,但主方法中的speed
变量的值并不会受到其影响。
4.2 引用类型的传值
引用类型参数的传递,调用方的变量,和接收方的参数变量,地址指向的是同一个对象。双方任意一方对这个对象的修改,都会影响对方。
我们来看一个实例:
运行结果:
- 1
- 2
- 3
- 4
- 5
由运行结果可见:方法replaceFirstPlayer
的参数players
是数组类型,也就是引用类型。当我们在方法内部对players
进行操作,其方法外部(即主方法中)的players
变量也受到了影响。不只是数组,对象也是引用类型,其参数传递同样遵循以上结论。我们在日常的编码中,请注意区分基本类型和引用类型的方法传值区别。
5. 可变参数
null我们可以使用可变参数列表声明方法的参数。可变参数列表的语法:
- 1
我们可以结合可变参数来实现一个求和方法sum
,请阅读下面实例:
运行结果:
- 1
- 2
- 3
上述实例中,在主方法中给sum
方法传参时,可选择一个或多个参数传递。方法体内对可变参数n
进行迭代,也可以将可变参数n
改为数组:int[] n
,得到的效果是相同的。
当方法的参数列表有两个或两个以上参数时,可变参数一定要放在最后,请查看如下实例:
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
另外,也可以将数组传递给可变参数列表,例如,在主方法中调用上面的search
方法:
运行结果:
- 1
- 2
上述两种传参方式都是合法的。
6. 方法重载
方法重载是指在一个类中定义多个同名的方法,但要求每个方法具有不同的参数的类型或参数的个数。调用重载方法时,Java 编译器能通过检查调用的方法的参数类型和个数选择一个恰当的方法。方法重载通常用于创建完成一组任务相似但参数的类型或参数的个数或参数的顺序不同的方法。
6.1 自定义方法的重载
例如,在Student
类中,有多个study
方法:
运行结果:
- 1
- 2
- 3
代码中的三个study
都是重载方法。通常来说,方法重载的返回值类型都是相同的。
如果我们在Student
类中再增加一个方法:
- 1
- 2
- 3
注意,上述的方法不是重载方法,因为我们已经在Student
类中定义了无参方法study
。
判断一个方法是否是重载方法的原则:方法名相同,参数类型或参数个数不同。
6.2 JDK 中的方法重载
Java
语言本身的类也定义了很多方法重载的例子,例如String
类的substring
方法,用于字符串截取:
- 1
- 1
如下为实际应用的实例:
- 1
- 2
- 3
- 4
- 5
运行结果:
- 1
- 2
7. 小结
本小节我们学习了 Java 方法,内容较多,我们一起来回顾一下所学的内容:
理解方法是一个命名语句块很重要。方法提高了代码的复用性,优质的方法提供者(如 JDK 中的方法),提高了调用者的开发效率。
定义方法时,要理解返回类型、方法名、参数列表和方法体的概念。通过方法的名称 + 括号运算符()
可以调用该方法。
方法内部遇到return
时返回,返回类型为void
表示不返回任何值。
对于方法传值问题,基本数据类型传值:方法中对参数的修改,不会影响到方法调用方传递的变量值;引用数据类型传值:方法中对参数的修改,会影响到方法调用方传递的对象。
可变参数通常用于参数数量是不确定的、可变化的方法。方法重载提高了程序的兼容性和易用性,为方法提供了多种可能性。