橘子味的心
标题:Java方法重载

如果一个类中有多个具有相同名称但参数不同的方法,则称为方法重载。如果只需要执行一个操作,具有相同的方法名称将增加程序的可读性。

假设必须执行给定数值的添加操作(求和),但是参数的数量不固定,如果为两个参数编写add1(int,int)方法,为三个参数编写add2(int,int,int)方法,可能对其他程序员来说很难理解这样的方法的行为,因为它的名称不同。

因此,执行方法重载能比较容易理解程序含义。

方法重载的优点

  • 方法重载提高了程序的可读性。

重载方法不同的方式

在java中重载方法有两种方式,它们分别是:

  • 通过改变参数的数量
  • 通过更改数据类型

注意:在java中,只通过更改方法的返回类型来实现方法重载是不可以的。

1. 方法重载: 修改参数的数量

在这个例子中,我们将创建了两个方法,第一个add()方法执行两个数值的相加操作,第二个add()方法执行三个数值的相加操作。

在这个例子中,我们创建静态方法,所以不需要创建实例来调用方法。

  1. class Adder {
  2. static int add(int a, int b) {
  3. return a + b;
  4. }
  5.  
  6. static int add(int a, int b, int c) {
  7. return a + b + c;
  8. }
  9. }
  10.  
  11. class TestOverloading1 {
  12. public static void main(String[] args) {
  13. System.out.println(Adder.add(11, 11));
  14. System.out.println(Adder.add(11, 11, 11));
  15. }
  16. }
  17. Java

上面代码执行后,输出结果如下 -

2. 方法重载:更改参数的数据类型

在这个例子中,创建了两种数据类型不同的方法。 第一个add方法接收两个整数参数,第二个add方法接收两个双精度型参数。

  1. class Adder {
  2. static int add(int a, int b) {
  3. return a + b;
  4. }
  5.  
  6. static double add(double a, double b) {
  7. return a + b;
  8. }
  9. }
  10.  
  11. class TestOverloading2 {
  12. public static void main(String[] args) {
  13. System.out.println(Adder.add(11, 11));
  14. System.out.println(Adder.add(12.3, 12.6));
  15. }
  16. }
  17. Java

上面代码执行后,输出结果如下 -

问题:为什么方法重载不能通过更改方法的返回类型?

在java中,只通过改变方法的返回类型来实现方法重载是不可能的,因为这样具有模糊性。 让我们来看看模糊性是怎么样发生的:

  1. class Adder {
  2. static int add(int a, int b) {
  3. return a + b;
  4. }
  5.  
  6. static double add(int a, int b) {
  7. return a + b;
  8. }
  9. }
  10.  
  11. class TestOverloading3 {
  12. public static void main(String[] args) {
  13. System.out.println(Adder.add(11, 11));// ambiguity
  14. }
  15. }
  16. Java

上面代码执行后,输出结果如下 -

  1. Compile Time Error: method add(int,int) is already defined in class Adder
  2. Java

System.out.println(Adder.add(11,11));//这里,java如何确定应该调用哪个sum()方法?

注意:编译时错误优于运行时错误。 所以,如果你声明相同的方法具有相同的参数,java编译器渲染编译器时间错误。

可以重载java main()方法吗?

这是完全可以的。 可以通过方法重载在类中有任意数量的main方法。 但是JVM调用main()方法,它只接收字符串数组作为参数。 让我们来看一个简单的例子:

  1. class TestOverloading4{
  2. public static void main(String[] args){System.out.println("main with String[]");}
  3. public static void main(String args){System.out.println("main with String");}
  4. public static void main(){System.out.println("main without args");}
  5. }
  6. Java

上面代码执行后,输出结果如下 -

  1. main with String[]
  2. Java

方法重载和类型提升

如果没有找到匹配的数据类型,那么会隐式地将一个类型提升到另一个类型。 让我们通过下面的图示来理解这个概念:

如上图所示,byte可以被提升为shortintlongfloatdoubleshort数据类型可以提升为intlongfloatdoublechar数据类型可以提升为intlongfloatdouble等等。

使用类型提升方法重载的示例

  1. class OverloadingCalculation1 {
  2. void sum(int a, long b) {
  3. System.out.println(a + b);
  4. }
  5.  
  6. void sum(int a, int b, int c) {
  7. System.out.println(a + b + c);
  8. }
  9.  
  10. public static void main(String args[]) {
  11. OverloadingCalculation1 obj = new OverloadingCalculation1();
  12. obj.sum(20, 20);// now second int literal will be promoted to long
  13. obj.sum(20, 20, 20);
  14.  
  15. }
  16. }
  17. Java

上面代码执行后,输出结果如下 -

如果找到匹配使用类型提升方法重载示例
如果方法中存在匹配类型参数,则不会执行类型提升方法重载。

  1. class OverloadingCalculation2 {
  2. void sum(int a, int b) {
  3. System.out.println("int arg method invoked");
  4. }
  5.  
  6. void sum(long a, long b) {
  7. System.out.println("long arg method invoked");
  8. }
  9.  
  10. public static void main(String args[]) {
  11. OverloadingCalculation2 obj = new OverloadingCalculation2();
  12. obj.sum(20, 20);// now int arg sum() method gets invoked
  13. }
  14. }
  15. Java

上面代码执行后,输出结果如下 -

  1. Output:int arg method invoked
  2. Java

在模糊的情况下使用类型提升来实现方法重载的示例

如果在方法中没有匹配的类型参数,并且每个方法都会提升相同数量的参数,那么会出现歧义。如下示例代码,将会产生编译时错误。

  1. class OverloadingCalculation3 {
  2. void sum(int a, long b) {
  3. System.out.println("a method invoked");
  4. }
  5.  
  6. void sum(long a, int b) {
  7. System.out.println("b method invoked");
  8. }
  9.  
  10. public static void main(String args[]) {
  11. OverloadingCalculation3 obj = new OverloadingCalculation3();
  12. obj.sum(20, 20);// now ambiguity
  13. }
  14. }
  15. Java

注意: 只有一种类型不被隐式地去提升,例如double不能被隐含地提升到任何类型。