Fork me on GitHub

Java中方法重写和方法重载的6个区别?

面试合集:https://gitee.com/mydb/interview

方法重写(Override)和方法重载(Overload)都是面向对象编程中,多态特性的不同体现,但二者本身并无关联,它们的区别犹如马德华之于刘德华的区别,除了名字长得像之外,其他的都不像。
image.png
接下来咱们就来扒一下二者的具体区别。

1.方法重写

方法重写(Override)是一种语言特性,它是多态的具体表现,它允许子类重新定义父类中已有的方法,且子类中的方法名和参数类型及个数都必须与父类保持一致,这就是方法重写。

1.1 基本用法

方法重写的基本用法如下:

/**
 * 父类中有一个 method 方法
 */
class Father {
    public void method(String name) {
        System.out.println("Father:" + name);
    }
}
/**
 * 子类中重写了父类的 method 方法
 */
class Son extends Father {
    @Override
    public void method(String name) {
        // 子类中重新定义了打印的行为,不再是 Father:XXX,而是 Son:XXX
        System.out.println("Son:" + name);
    }
}

1.2 使用场景

方法重写在 Java 中应用的场景有很多,其中比较经典的是 Object 类中的 equals 方法。
Object 是所有类的父类,equals 方法在 Object 类中的实现源码如下:
image.png
从上述源码可以看出,Object 类中的 equals 是用来比较两个对象的引用地址是否相同的。
但对于字符串来说,它是被 final 修饰的,所以每次创建的字符串引用地址都不同,那这时比较引用地址就没意义了,所以 String 类必须重写父类 Object 中的 equals 方法,具体源码如下:
image.png
通过上述代码我们可以看出,String 中的 equals 方法,已经不是比较两个字符串的引用地址是否相同了,而是判断两个字符串的值是否相同。

1.3 注意事项

在方法重写时需要注意的事项有以下 5 个:

  • 注意事项 1:子类方法的权限控制符不能变小,也就是如果父类方法的权限控制符是 protected,那么子类的方法权限控制符只能是 protected 或 public;
  • 注意事项 2:子类方法返回的类型只能变小,也就是说如果父类方法返回的是 Number 类型,那么子类方法只能返回 Number 类型或 Number 类的子类 Long 类型,而不能返回 Number 类型的父类类型 Object;
  • 注意事项 3:子类抛出异常的类型只能变小;
  • 注意事项 4:子类方法名必须和父类方法名保持一致;
  • 注意事项 5:子类方法的参数类型和个数必须和父类保持一致。

2.方法重载

方法重载是指在同一个类中,定义了多个同名方法,但同名方法的参数类型或参数个数不同就是方法重载。

2.1 基本使用

以下 4 个 method 方法就是方法重载:

public class OverloadExample {
    public static void method() {
        // doSomething
    }

    public static void method(String name) {
        // doSomething
    }

    public static void method(Integer id) {
        // doSomething
    }

    public static void method(Integer id, String name) {
        // doSomething
    }
}

2.2 使用场景

方法重载的经典使用场景是 String 类型的 valueOf 方法,valueOf 方法重载有 9 种实现,如下图所示:
image.png
它可以将数组、对象和基础数据类型转换成字符串类型。

2.3 注意事项

方法重载的优先匹配原则有以下 5 个:

  • 第 1 匹配原则:优先匹配相同数据类型,方法重载会优先调用和方法参数类型一模一样的方法,比如只有一个 String 类型的参数调用,会优先匹配只有一个 String 参数类型的重载方法;
  • 第 2 匹配原则:如果是基本数据类型,会自动转换成更大的基本数据类型进行匹配,比如调用的参数是 int 类型,那么会优先调用基本类型 long,而非包装类型 Integer(这点需要特殊注意,容易出错)。
  • 第 3 匹配原则:自动装箱和自动拆箱匹配,参数调用也会进行自动拆箱和自动装箱的方法匹配,比如调用参数传递的是 int 类型,那么它可以匹配到 Integer 类型的重载方法;
  • 第 4 匹配原则:会按照继承路线依次向上匹配父类,如果匹配不到当前类,会尝试匹配它的父类,或者是父类的父类,依次往上匹配;
  • 第 5 匹配原则:可变参数匹配,如果方法是可选参数方法,那么它的调用优先级是最低的,在最后阶段才会匹配可选参数方法。

方法重载会按照以上的 5 个原则依次进行匹配,符合规则的方法会被优先调用。除了以上匹配原则之外,还需要特殊注意一点,不同的返回类型不能作为方法重载的依据,也就是不同的返回值类型不算方法重载,关于这一点的详细介绍可参考:https://mp.weixin.qq.com/s/4pi1OZx8So6GjHD6yxjB3Q

3.方法重写 VS 方法重载

方法重写和方法重载的具体不同,主要体现在以下 6 个方面:

区别 方法重写 方法重载
方法关系 描述的是父类和子类的方法关系 描述的是同一个类中多个同名方法的方法关系
Override 关键字 需要使用 @Override 关键字修饰 不需要
参数类型和参数个数 不能修改 必须修改
返回类型 只能变小或相同,不能变大 可修改
抛出异常 只能变小或删除,不能抛出新的或者更大的异常 可修改
权限控制符 只能变大,不能变小 可修改

总结

方法重写(Override)和方法重载(Overload)都是面向对象编程中,多态特性的不同体现,方法重写描述的是父类和子类的方法关系,而方法重载描述的是同一个类中多个同名方法的方法关系。除此之外方法重写和方法重载还有:Override 关键字、参数类型和参数个数、返回类型、抛出异常和权限控制符等不同点。

是非审之于己,毁誉听之于人,得失安之于数。

公众号:Java面试真题解析

posted @ 2022-01-11 09:25  磊哥|www.javacn.site  阅读(1620)  评论(0编辑  收藏  举报