【Java15】继承

继承是面向对象三大特征之一,也是软件代码服用的重要手段。

Java只允许单继承,即每个子类只有一个直接父类。

C++中的多继承被Java舍弃了,原因是多继承一方面难以准确表述类之间的关系,另一方面很容易造成代码错误。总结起来就两个字:”难用“。

Java的继承有一个关键字extends,实现继承的叫子类,被继承的类叫父类(也叫基类、超类)。从面向对象的角度,父类是更一般的概念,子类是更特殊的概念,例如”水果之于苹果“。

Java继承的通用代码框架为:

修饰符 class SubClass extends SuperClass
{// 类定义
}

extends这个词翻译为”扩展“,这个含义其实更准确:子类是对父类的扩展,子类是父类的定制化、特殊化。

从”继承“的角度,子类继承了什么呢?父类的全部成员变量、方法和内部类(包括内部接口、枚举)。

从”扩展“的角度,子类是父类的扩充,所以当然拥有父类的全部内容。

但是,子类唯独不能获得父类的构造器

public class Fruit
{public double weight;public void info(){System.out.println(weight);}
}public class Apple extends Fruit
{public static void main(String[] args){var a  = new Apple();// Apple对象本身没有weight,但是它是Fruit的扩展,所以就拥有了weight这个成员变量System.out.println(a.weight);// 同时也继承了info这个方法a.info();}
}

注意Java继承和C++的区别

  • Java摒弃了多继承
  • Java类只能有一个直接父类,但是可以有多个间接父类(类似于数据结构中树的父节点和祖先节点)

如果一个Java类没有显示指定直接父类,默认扩展java.lang.Object(所有类的基类/父类)。

因此,可以这么说:java.lang.Object要么是一个类的直接父类,要么是它的间接父类

重写

大多数情况下,子类全盘接受父类的方法就行了。但是有时候,根据需求需要重写父类的方法。这时,子类和父类中出现了同名的方法,这种现象称为重写(Override),也翻译为”覆盖“:子类的方法覆盖了父类的方法。

重载类似但有所不同,重写需要遵循”两同、两低、一高“规则:

  • 方法名相同,形参列表相同(两同);
  • 子类方法的返回值类型比父类方法的返回值类型低或相等;子类方法声明抛出的异常类应比父类方法声明抛出的异常类低或相等;
  • 子类方法的访问权限要比父类方法的访问权限更高或相等。

无论是被覆盖的方法,还是覆盖的方法,都只能是实例方法,不能是类方法

想一想为什么?主要是从内存分配的角度来看的。

class BaseClass
{public static void test(){}
}class SubClass extends BaseClass
{public void test(){} // 错误,子类试图覆盖一个类方法
}

子类覆盖了父类的方法后,**子类的对象不能再调用父类被覆盖的方法了。**但是,在子类的方法中,可以使用super关键字来调用父类的方法(实例方法),或者直接使用父类类名.类方法来调用(类方法)被覆盖的方法。

如果父类的方法设置了private权限,子类本身就无法访问该方法(注意:即便是父类和子类的关系,父类也可以利用private来隐藏一些内容)。继续无法访问,那么即使子类中定义了一个和父类同名、同形参列表的方法,也不算是重写,而是普通的一个新方法而已:

class BaseClass
{private void test(){} // private权限
}class SubClass extends BaseClass
{public static void test() // 这是一个全新的方法,所以可以声明为类方法
}
super关键字

(1)在子类的方法中可以使用super来调用父类被覆盖的方法:

public class Bird
{public void fly(){System.out.println("我在天空自在地飞翔");}
}// 鸵鸟
public class Ostrich extends Bird 
{// Overridepublic void fly(){System.out.println("我只能在地上奔跑")}public static void main(String[] args){var os = new Ostrich();// 执行鸵鸟类的实例方法os.fly();}public void callOverrideMethod(){super.fly(); // 利用super调用父类中被覆盖的实例方法}
}

注意:this不能出现在static修饰的方法中。因为this调用的只能是实例方法;同理super调用的也只能是实例方法。如果它们位于static修饰的方法(即类方法)中,那这个方法可能被类调用。这时,thissuper都没有定义了。所以不允许它们出现在类方法中。

(2)在子类的方法中也可以使用super来访问父类被覆盖的实例变量。和方法重写类似,子类中定义的与父类同名的实例变量会隐藏父类的实例变量。通过super能够访问父类中被隐藏的实例变量。

class BaseClass
{public int a = 5;
}public class SubClass extends BaseClass
{public int a = 7;public void accessOwner(){System.out.println(a);}public void accessBase(){// super不能出现在static方法中System.out.println(super.a);}public static void main(String[] args){var sc = new SubClass();sc.accessOwner(); // 7sc.accessBase(); 	// 5}
}

如果子类没有和父类同名的实例变量,直接访问父类的变量即可,不需要使用super或父类的类名。对没有显式制定调用的变量,系统会按照如下顺序查找这个变量的来源:

  1. 该方法中有没有叫该名字的局部变量;
  2. 当前类中是否包含叫该名字的成员变量;
  3. 当前类的直接父类中是否包含该名字的成员变量。如果没有,就向上回溯,直到java.lang.Object
  4. 如果都没有找到,就会报编译错误。

如果被覆盖的是类变量,那就通过父类名.类变量来访问父类的类变量。

Java中,创建一个子类对象时,不单会为该类中定义的实例变量分配内存,也会为它从父类(直接父类、间接父类)继承的所有实例变量分配内存,即使它的实例变量和父类重名。

例如,一个类有两个父类(直接父类A,间接父类B),A中有两个实例变量,B中有3个实例变量,当前类有2个实例变量,那么一共会分配7个实例变量的空间。

因此,所谓的变量隐藏,依然会为父类中被隐藏的实例变量分配内存空间。

调用父类的构造器

子类不会获得父类的构造器,但是可以调用父类构造器的初始化代码,类似前面使用this来在构造器中调用重载的构造器,只不过这里用super来完成。

class Base
{public double size;public String name;public Base(double size, String name){this.size = size;this.name = name;}
}public class Sub extends Base
{public String color;public Sub(double size, String name, String color){// 使用super来复用父类的构造器中的代码super(size, name);this.color = color;}
}

this调用重载构造器代码类似,super调用父类构造器代码也要位于当前类构造器的第一行。

无论是否使用super,子类构造器总会调用父类构造器一次:

  1. 第一种情况,子类构造器第一行使用super显式调用父类构造器。就是上面的情况。
  2. 第二种情况,子类没有使用super,这是系统在执行子类构造器前会隐式调用父类的无参构造器。
  3. 第二种情况,子类构造器使用this调用本类另一个构造器,但是执行另一个构造器的时候,可能遇到1或者2的情况。

上述调用方式会不断上溯,直到到达java.lang.Object,也就是说,最先执行的永远是java.lang.Object类的构造器。

对于下面的继承关系:

在这里插入图片描述

class Creature()
{public Creature(){System.out.println("Creature无参构造器");}
}class Animal extends Creature
{public Animal(String name){System.out.println("Animal带一个参数的构造器");}public Animal(String name, int age){this(name); // 使用this调用重载的构造器System.out.println("Animal带两个参数的构造器");}
}public class Wolf extends Aminal
{public Wolf(){// 使用super调用Animal带两个参数的构造器super("灰太狼", 3);System.out.println("Wolf无参数构造器");}public static void main(String[] args){new Wolf();}
}

程序执行的结果应该是:

Creature无参数构造器
Animal带一个参数的构造器
Animal带两个参数的构造器
Wolf无参数的构造器

创建任何对象都是从该类所在继承树的根所在的类的构造器开始执行,依次向下。如果遇到this,则依次执行该层所在类的其他构造器。把上面的图换一种画法:

在这里插入图片描述

”万物之源“java.lang.Object类一直存在,但是没有存在感。因为他只有一个默认的构造器,且构造器里不输出任何内容。因此,我们感受不到调用它的构造器。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/43236.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

双系统ubuntu20.04扩容

windows端 打开磁盘管理器,选择需要的盘点击压缩卷 点击未分配的盘,新建简单卷,一致点击下一步即可,记住分配的大小容量 ubuntu端 lsblk 查看所有的磁盘,可以看到新增为nvme0n1p4、nvme1n1p2 win分配的格式为NTFS&a…

【Excel】 批量跳转图片

目录标题 1. CtrlA全选图片 → 右键 → 大小和属性2. 取消 锁定纵横比 → 跳转高度宽度 → 关闭窗口3. 最后一图拉到最后一单元格 → Alt吸附边框![](https://i-blog.csdnimg.cn/direct/d56ac1f41af54d54bb8c68339b558dd1.png)4. CtrlA全选图片 → 对齐 → 左对齐 → 纵向分布!…

全网最适合入门的面向对象编程教程:11 类和对象的Python实现-子类调用父类方法-模拟串口传感器和主机

全网最适合入门的面向对象编程教程:11 类和对象的 Python 实现-子类调用父类方法-模拟串口传感器和主机 摘要: 本节课,我们主要讲解了在 Python 类的继承中子类如何进行初始化、调用父类的属性和方法,同时讲解了模拟串口传感器和…

OpenHarmony 入门——单元测试UnitTest快速入门

引言 OpenHarmony 的单元测试(UnitTest)是一个关键的软件开发过程,它确保代码的各个部分能够按预期工作,OpenHarmony的测试框架中提供了很多种的单元测试,今天简单介绍下UnitTest 类型的TDD测试。 OpenHarmony 的TDD …

Nacos 国际化

项目需要,后端异常信息需要进行国际化处理。所有想有没有方便易用的可选项。 1、国际化配置调整,不需要重启系统 2、可支持添加不同或自定义语言包(就是配置的资源文件) 参考: Nacos实现SpringBoot国际化的增强_spr…

LeetCode热题100刷题9:25. K 个一组翻转链表、101. 对称二叉树、543. 二叉树的直径、102. 二叉树的层序遍历

25. K 个一组翻转链表 /*** Definition for singly-linked list.* struct ListNode {* int val;* ListNode *next;* ListNode() : val(0), next(nullptr) {}* ListNode(int x) : val(x), next(nullptr) {}* ListNode(int x, ListNode *next) : val(x), nex…

go语言day12 包 init() time包 file包

包 包中的 结构体 及结构体属性 方法 都可以通过设置首字母大小写来实现权限访问,首字母大写任何包中都可以访问,首字母小写只在同包中可以访问。 再导入包go文件时,可以给.go文件取别名。 在导入的包名前加入 _ 意思是调用该包的初始…

普中51单片机:矩阵按键扫描与应用详解(五)

文章目录 引言电路图开发板IO连接矩阵键盘的工作原理行列扫描逐行/逐列扫描 LCD1602代码库代码演示——暴力扫描代码演示——数码管(行列式)代码演示——线翻转法代码演示——LCD1602密码锁 引言 矩阵按键是一种通过行列交叉连接的按键阵列,可以有效地减少单片机I/…

计算机网络 - 万字长文

计算机网络 二、计算机网络2.1 七层模型表格2.2 通俗讲解七层模型2.3 TCP与UDP对比2.4 TCP 三次握手过程==为什么握手是三次,而不是两次或者四次?====三次握手可以携带数据吗?====TCP三次握手失败,服务端会如何处理?====什么是半连接队列?全连接====ISN(Initial Sequence…

基于单片机的太阳能热水器控制系统设计

随着我国经济水平的不断提高,民众对生活质量的追求也在不断进步,对于现代家庭而言,热水器成为了必备的生活电器。目前市面上的电器主要是电热水器、燃气热水器以及太阳能热水器。就能源节约性能而言,太阳能热水器占据了绝对优势&a…

Java | Leetcode Java题解之第223题矩形面积

题目: 题解: class Solution {public int computeArea(int ax1, int ay1, int ax2, int ay2, int bx1, int by1, int bx2, int by2) {int area1 (ax2 - ax1) * (ay2 - ay1), area2 (bx2 - bx1) * (by2 - by1);int overlapWidth Math.min(ax2, bx2) -…

开源高效在线电子表格解决方案:Luckysheet

Luckysheet:体验幸运,掌握高效数据表格编辑!- 精选真开源,释放新价值。 概览 Luckysheet 是一个功能强大、配置简单且完全开源的在线电子表格工具,它类似于我们熟知的 Excel,但更加灵活和易于集成。它是一…

windows环境下部署多个端口Tomcat服务和开机自启动设置保姆级教程

前言 本文主要介绍了 windows环境下,配置多个Tomcat设置不同端口启动服务。其实在思路上Linux上也是适用的,只是 Linux 上没有可视化客户端,会麻烦些,但总体的思路上是一样的。 注:文章中涉及些文字和图片是搬运了其他…

OpenGL3.3_C++_Windows(28)

Demo演示 demo 视差贴图 视差/高度/位移贴图(黑--白):和法线贴图一样视差贴图能够极大提升表面细节,使之具有深度感。第一种思路(置换顶点):对于一个quad ,分成约1000个顶点&#x…

C语言 | Leetcode C语言题解之第223题矩形面积

题目: 题解: int computeArea(int ax1, int ay1, int ax2, int ay2, int bx1, int by1, int bx2, int by2) {int area1 (ax2 - ax1) * (ay2 - ay1), area2 (bx2 - bx1) * (by2 - by1);int overlapWidth fmin(ax2, bx2) - fmax(ax1, bx1), overlapHei…

Visual Studio Code 教程 VsCode安装Live Server以服务形式打开html

搜索Live Server 插件,然后安装 选一个html文件,右键点击 Open with live server,然后就自动弹出来了

使用paddleOCR训练自己的数据集到ONNX推理

一、环境安装 1、安装paddlepaddle; https://www.paddlepaddle.org.cn/ 这里安装2.6.1的话使用onnx会出现swish算子报错的问题 python -m pip install paddlepaddle-gpu2.5.2 -i https://pypi.tuna.tsinghua.edu.cn/simple验证是否成功安装 python import paddl…

Mysql Workbench的使用

本篇内容:对Mysql Workbench的常规使用学习 一、知识储备 1. Workbench 可以做什么 是mysql数据库可视化管理的一款免费工具,除了平常的通过sql语句,进行创建数据库表、增删改查外,还可以利用其进行建模创建数据库表。通过创建…

人工智能的新时代:从模型到应用的转变

💝💝💝欢迎来到我的博客,很高兴能够在这里和您见面!希望您在这里可以感受到一份轻松愉快的氛围,不仅可以获得有趣的内容和知识,也可以畅所欲言、分享您的想法和见解。 推荐:kwan 的首页,持续学…

【Linux】记录一起网站劫持事件

故事很短,处理也简单。权当记录一下,各位安全大大们手下留情。 最近一位客户遇到官网被劫持的情况,想我们帮忙解决一下(本来不关我们的事,毕竟情面在这…还是无偿地协助一下),经过三四轮“谦让…