40从零开始学Java之方法重写和方法重载到底有哪些区别?

news/2025/1/15 6:56:09/

作者:孙玉昌,昵称【一一哥】,另外【壹壹哥】也是我哦

千锋教育高级教研员、CSDN博客专家、万粉博主、阿里云专家博主、掘金优质作者

前言

在前面的两篇文章中,壹哥给大家讲解了面向对象中的封装和继承,现在还有一个多态没有给大家讲到。但是在学习多态之前,我们需要提前知道“方法重写”是怎么回事,因为方法重写是实现多态的一个必要条件。所以如果我们不知道方法重写是咋回事,学习多态也就会有点费劲。废话少说,直接上干货!

--------------------------------------------------前戏已做完,精彩即开始----------------------------------------------

全文大约【4000】字,不说废话,只讲可以让你学到技术、明白原理的纯干货!本文带有丰富的案例及配图视频,让你更好地理解和运用文中的技术概念,并可以给你带来具有足够启迪的思考......

配套开源项目资料

Github:

GitHub - SunLtd/LearnJava

Gitee:

一一哥/从零开始学Java

一. 方法重写

在面向对象中,实现多态的必备条件是继承、重写和向上转型,现在我们已经学习了什么是继承。接下来我们再来学习什么是方法重写,这是我们能够实现多态的前提。

1. 概念

如果我们在子类中,创建了一个与父类中名称、返回值类型、参数列表都完全相同的方法,只是方法体的功能实现不同,这种方式被称为方法重写(override),或者叫方法覆盖。当父类中的方法无法满足子类的需求,或者子类需要有特殊功能时,就可以进行方法重写。

2. 基本要求

我们在进行方法重写时,需要遵循以下几点要求:

  • 父类的成员方法只能被它的子类重写,即不能继承一个方法,就不能重写这个方法;
  • 被final修饰的方法不能被重写;
  • 被static修饰的方法不能被重写,但可以再次声明;
  • 构造方法不能被重写;
  • 子类和父类在同一个包中时,子类可以重写父类中除了被private和final修饰的其他所有方法;
  • 子类和父类不在同一个包中时,子类只能重写父类被public和protected修饰的非final方法;
  • 重写的方法建议使用@Override注解来标识。

3. 注意事项

另外我们在进行方法重写时,还要注意以下几点:

  • 方法签名要相同:重写的方法和被重写的方法,在方法名、参数上都要相同;
  • 返回值类型一致:JDK 1.5之前重写方法的返回值类型必须一样,但之后的Java版本放宽了限制,返回值类型必须小于或等于父类方法的返回值类型
  • 访问修饰符要更宽泛:子类重写父类的方法时,子类方法中的访问修饰符不能比父类中的更严格(public>protected>default>private)。比如父类方法的修饰符是protected,则子类的同名方法其修饰符可以是protected或public,但不能是默认的或private;
  • 声明的异常类型要一致:重写的方法一定不能抛出新的检査异常,或者比被重写方法声明更宽泛的检査型异常。例如,父类的方法声明了IOException,重写该方法时就不能抛出Exception,只能拋出IOException或其子类异常,但可以抛出非检査异常

4. 代码实现

接下来壹哥就通过一个案例来给大家讲解方法的重写该怎么实现。

4.1 定义父类

我们先定义一个Father父类,要注意父类中有哪些方法不能被重写。

public class Father {// 父类中的成员变量--变量隐藏String name="老子";//构造方法不能被重写,因为构造方法不能被继承!public Father() {System.out.println("爹的构造方法");}// 吃public void eat() {System.out.println("爹吃馒头");}// 喝public void drink() {System.out.println("爹喝水");}// 玩//私有方法不能被重写
//	private void play() {
//		System.out.println("爹玩火");
//	}//静态方法不能被重写,但可以在子类中声明一个同样的静态方法。
//	public static void play() {
//		System.out.println("爹玩火");
//	}//final方法不能被重写public final void play() {System.out.println("爹玩火");}
}

4.2 定义子类

定义一个Son子类继承父类,有了继承才会有重写!

public class Son extends Father{//构造方法不能被重写,因为构造方法不能被继承!//@Override//public Father() {}// 吃/*** 方法重写时可以带有@Ovriride关键词,当重写的方法签名不一致时,可以有编译错误的提示。* 否则方法签名不一致时不会有错误提示,会被当做一个新的方法来处理。*/@Overridepublic void eat() {//如果子类的功能,是在父类的基础之上进行的额外扩展增加,//我们可以使用super关键字调用父类的同名方法,然后再进行自己的额外扩展!//如果子类的实现和父类完全不一样,可以不调用super!super.eat();//方法重写时,子类可以对父类的同名方法进行扩展实现,方法体的内容可以和父类中的实现不一样System.out.println("儿子吃肉");}// 喝@Overridepublic void drink() {//如果子类的实现和父类完全不一样,可以不调用super!System.out.println("儿子喝酒");}/*** 父类私有的、static、final等方法无法被重写*///@Override//public void play() {}//static静态的父类方法不能被重写,但可以在子类中再重新编写一个静态的同名方法。//public static void play() {}//变量隐藏--调用父类和子类中的同名成员变量public void sayHello() {// 如果子类的实现和父类完全不一样,可以不调用super!System.out.println("父亲的名字=" + super.name);System.out.println("儿子的名字=" + name);}public static void main(String[] args) {Son son = new Son();son.sayHello();}
}

我们在进行方法重写时,要注意以下几点:

  • 方法重写时可以带有@Ovriride关键词。当重写的方法签名不一致时,会有编译错误的提示,否则方法签名不一致时不会有错误提示,会被当做一个新的方法来处理。
  • 当子类对象调用重写的方法时,默认执行的是子类的方法,而不是父类中被重写的方法。如果我们想要调用父类中被重写的方法,则可以使用“super.方法名”的形式。
  • 如果子类的功能是在父类的基础之上进行的额外扩展,我们可以使用super关键字调用父类的同名方法,然后再进行自己的额外扩展!
  • 如果子类的实现和父类完全不一样,可以不调用super!
  • 方法重写时,子类可以对父类的同名方法进行扩展实现,方法体的内容可以和父类中的实现不一样。

4.3 @Override注解

在上面的代码中,我们用到了一个新的关键字@Override。在Java中,@Override是一个注解,关于注解的更多内容,壹哥会在后面的文章中进行专门讲解,现在我们先知道注解这个概念就行。

@Override是一个用来修饰被重新的方法的注解,只能用在被重新的方法上,不能用在其它的地方。该注解可以强制子类必须重写父类的方法或者接口中的方法,主要是告诉编译器检查重写的方法是否和父类中定义的一致。如果重写的方法签名不一致,会提示编译错误。如果方法签名不一致,则不会有错误提示,会被当做一个新的方法来处理。通过这样的机制,就可以避免程序员出现一些低级的错误。

5. 变量隐藏

5.1 概念

如果子类中定义了一个成员变量,而该变量的名称与父类中的成员变量相同,数据类型不一定完全一致,我们就把这称为变量隐藏。也就是说,子类的成员变量,对从父类继承过来的成员变量进行了重新定义,出现了子类变量对父类变量的隐藏。所以子类执行自己定义的方法时,操作的成员变量默认是自己定义的变量,而不是父类中的同名变量。如果我们非要操作隐藏的成员变量,可以使用super关键字进行调用。

接下来我们通过一个案例来给大家演示变量隐藏的使用。

5.2 案例实现

父类中定义一个成员变量name,如下图所示:

子类中也定义一个相同的成员变量name,如下图所示:

如果我们在Son类中直接使用name,默认使用的是Son自己的变量;如果我们想使用Father类中的name变量,则可以通过“super.属性”的形式进行。执行结果如下图所示:

6. 方法隐藏

在子类继承父类时,既然存在变量隐藏的现象,同理也存在方法隐藏的现象。

6.1 概念

我们知道,方法的重写是子类覆盖父类的对象方法,而方法隐藏则是子类覆盖父类的静态方法(类方法)。在java中的静态方法能被子类继承吗?答案是肯定的,但若子类中有与父类中同名同参的方法,则父类的方法将被隐藏。

6.2 案例实现

我们先定义一个Father父类,里面有个静态方法eat。

/*** @author 一一哥Sun * 千锋教育* 定义父类*/
public class Father {// 吃---静态方法public static void eat() {System.out.println("爹吃馒头");}
}

然后再定义一个Son子类,里面也有一个静态方法eat。我们知道,静态方法是可以被继承的,所以如果Son子类中没有定义自己的eat()方法,默认可以使用Father父类中的eat()方法。但如果我们在子类中也定义了一个eat()方法,子类的同名静态方法就会隐藏父类中的eat()方法,这就是方法隐藏。

/*** @author 一一哥Sun * 千锋教育* 子类继承父类*/
public class Son extends Father {// 吃---静态方法//如果子类中没有定义该方法,则子类可以继承使用父类的eat()方法public static void eat() {//子类覆盖父类中的同名静态方法(类)System.out.println("儿子吃肉");}public static void main(String[] args) {//调用子类自己的静态方法eat();//调用父类的静态方法Father.eat();}
}

执行结果如下图所示:

6.3 小结

通过本案例,我们可以得出以下结论:

  1. 子类可以继承父类中的静态方法;
  2. 子类无法重写父类中的静态方法,但可以重载;
  3. 若子类中定义了与父类中同样的静态方法,则父类的同名方法会被子类隐藏

二. 重写与重载的区别

在Java中既有方法重写(Override),也有方法重载(Overload),对于初学者来说很容易搞混。所以有不少面试官,在招聘初级程序员时,就很喜欢问我们方法重写与方法重载的区别。

其实方法重写Override和方法重载Overload的最大不同,在于方法签名的不同。如果同一个类中的多个方法签名不同,就是方法重载Overload,重载出的方法是一个新方法。如果父子类之间的多个方法签名相同,且返回值也相同,就是方法重写Override。

当然,如果你想把关于重写和重载的区别说得更详细,可以参考以下章节。

1. 重载的特点

  • 方法重载要求方法同名不同参(参数类型、个数、顺序);
  • 重载的方法与返回值、访问修饰符无关;
  • 重载的方法发生在同一个类中,是在一个类中创建多个同名的方法。

2. 重写的特点

  • 重写的方法发生在父子类中,需要有继承关系;
  • 父类的成员方法只能被它的子类重写,即不能继承一个方法,就不能重写这个方法;
  • 被final修饰的方法不能被重写;
  • 被static修饰的方法不能被重写,但可以再次声明;
  • 构造方法不能被重写;
  • 子类和父类在同一个包中时,子类可以重写父类中除了被private和final修饰的其他所有方法;
  • 子类和父类不在同一个包中时,子类只能重写父类被public和protected修饰的非final方法;
  • 方法重写时可以使用@Override注解;
  • 方法签名要相同
  • 返回值类型一致
  • 访问修饰符要更宽泛
  • 声明的异常类型要一致

------------------------------------------------​​​​​​​正片已结束,来根事后烟----------------------------------------------

三. 结语

现在你知道方法重写是怎么回事了吗?另外方法重载和方法重写的区别,是我们面试初级程序员时很常见的题目,大家一定要牢牢掌握哦。现在有了方法重写的基础,接下来我们就可以学习多态的内容了,敬请关注下一篇文章哦。

另外如果你独自学习觉得有很多困难,可以加入壹哥的学习互助群,大家一起交流学习。

四. 配套视频

如果你不习惯阅读技术文章,或是对文中的技术概念不能很好地理解,可以来看看壹哥帮你筛选出的视频教程。与本文配套的Java学习视频,链接如下:

Bilibili External Player

五. 今日作业

1. 第一题

设计一个父类和一个子类,让子类超越父类的能力。


http://www.ppmy.cn/news/256939.html

相关文章

SATA电源线的作用

普通硬盘和光驱的电源接口为D型口,由4芯组成;其中,红线所对应的5V电压输入,黄线对应输出的是12V电压,两条黑线为接地;5芯SATA电源线多了一组+3.3V,可以减少硬盘内部供电二次转换时带…

图解SSD不同的接口:SATA、MSATA、M.2 SATA、M.2 NVME

视频讲解HDD, SSD, SATA、MSATA、M.2 SATA、M.2 NVME:https://www.bilibili.com/video/BV1hp4y1z76t -------------------------------------------- SSD,固态硬盘,区别于老旧的机械硬盘(HDD)。 目前常见的有三种: SATA MSAT…

【转】【详细】SATA、mSATA 、PCIe和M.2——SSD硬盘的接口

犹记得当年Windows 7系统体验指数中,那5.9分磁盘分数,在其余四项的7.9分面前,似乎已经告诉我们机械硬盘注定被时代淘汰。势如破竹的SSD固态硬盘,彻底打破了温彻斯特结构的机械硬盘多年来在电脑硬件领域的统治。SSD数倍于HDD机械硬…

SATA硬盘的数据和电源接口定义

现在 SATA设备越来越普及,包括STAT硬盘和光驱基本都已经是 SATA接口的了,以前的老式电源输出接口一般都是20针供主板加上4针的电源供硬盘也就是说以前的电脑电源给硬盘供电没有设计15针 SATA接口的供电,现在的 SATA电源是15针的,那…

SATA 3.0 双通道 - 硬盘数据线 + 硬盘电源线

SATA 3.0 双通道 - 硬盘数据线 硬盘电源线 1. 硬盘数据线 硬盘电源线 2. 兼容 SATA 接口 3. 提供直头和弯头可选 4. 6 Gbps 速率 5. 铝箔屏蔽 6. SATA 接口对照 7. 数据线/电源线 连接方法

SATA接口定义

SATA接口定义 SATA 是Serial ATA的缩写,即串行ATA。2001年,由Intel、APT、Dell、IBM、希捷、迈拓这几大厂商组成的Serial ATA委员会正式确立了Serial ATA 1.0规范,2002年确立了Serial ATA 2.0规范。Serial ATA采用串行连接方式,串…

数据的IO和复用

文章目录 1. IO 函数1.1 使用recv()函数接收数据1.2 使用send()函数发送数据1.3 使用readv()函数接收数据1.4 使用writev()函数发送数据1.5 使用recvmsg()函数接收数据1.6 使用sendmsg()函数发送数据 2. select()函数和pselect()函数2.1 select()函数2.2 pselect()函数 3. poll…

免费的配音软件--- tts-vue 软件 下载安装过程

视频效果 tts-vue 软件 图片效果 软件包含有; 语言: 高棉语(柬埔寨) 马耳他语(马耳他) 马来语(马来西亚) 马拉雅拉姆语(印度) 马拉地语(印度) 马其顿语(北马其顿) 韩语(韩国) 阿拉伯语(黎巴嫩) 阿拉伯语(阿昙) 阿拉伯语(阿拉伯联合酋长国) 阿拉伯语(阿尔及利亚) 阿拉伯语(约…