当前位置: 首页 > news >正文

03、继承与多态

本章跟上一节性质类似,比较繁,需要自己手动多多复盘和总结


一、啥是继承?

1、继承的定义

主要意思就是对代码进行共性抽取来实现代码复用

public class java0608  {static String name ="dog";static int age = 5;static void bark(){System.out.println("汪汪");}}
public class java0607 extends java0608 {static int a = 1;public static void main(String[] args) {System.out.println(name);java0608.bark();}
}

这两段代码就是一个典型的继承,通过子类+extends+父类的形式得到了父类的变量和方法

注意:子类要有不同于父类的成员,否则就没必要继承了

2、子类与父类成员变量同名

在访问成员变量时以子类变量优先,假如恰好子类变量和父类变量同名,要访问父类的变量则需要在前面加上super用以区分

简单概括一下就是:子类没有父类找,父类没有就报错

这里我分成三段代码便于阅读:

public class java0608  {public int a = 9;public int b = 3;public int c = 5;
}
public class java0607 extends java0608 {public int a = 1;public void test(){System.out.println(super.a);System.out.println(this.b);System.out.println(c);}
}
public class java060702 {public static void main(String[] args) {java0607 t =new java0607();t.test();}
}

3、子类方法与父类同名

这个跟上面也是同理,这里需要特别注意的是:

一旦父类与子类方法同名,也就构成了我们常见的重载,这个时候就有了两种匹配情况:

1、根据你所调用方法后面括号里传入的参数来给你匹配合适的方法

2、参数个数都一样就仍然是子类优先

public class java0608  {public int a = 9;public int b = 3;public int c = 5;public void method(){System.out.println("调用了父类的方法");
}
public class java0607 extends java0608 {public int a = 1;public void test(){System.out.println(super.a);System.out.println(this.b);System.out.println(c);super.method();}public void method(){System.out.println("调用了子类的方法");}
}

其实到这里我们是不是就发现super跟this有点像?

对,我把它俩列举出来:

this.data访问当前类的成员变量          super.data访问父类的成员变量
this.fun()访问当前类的成员方法         super.fun()访问父类的成员方法 
this()      访问当前类的构造方法         f super()      访问???(引出下文) 

所以它们都是只能在非静态方法中使用,且只能在第一行,并且他俩不能同时出现

二、super访问构造方法

当在子类中使用构造方法来初始化变量时,我们往往会在他完成构造之前先帮助父类对其中的成员进行初始化

举个例子:

public class java0608  {
//
public java0608(){System.out.println("调用了父类的构造方法");
}
public class java0607 extends java0608 {public java0607(){
//          super();    当⽗类的构造⽅法是不带参数的构造⽅法且只有这⼀个的情况下,默认会添加⼀个super();System.out.println("调用了子类的构造方法");}
public class java0609 {public static void main(String[] args) {java0607 t =new java0607();
//    t.test();
}
}

这个运行的结果就是先父类后子类的形式

三、代码块的访问顺序

public class java0607 extends java0608 {static {System.out.println("访问了子类的静态代码块1");}{System.out.println("访问了子类的实例代码块2");}public java0607(String name,int age){super(name,age);    //当⽗类的构造⽅法是不带参数的构造⽅法且只有这⼀个的情况下,默认会添加⼀个super();System.out.println("访问了子类的构造代码块3");}
public class java0608  {static {System.out.println("访问了父类的静态代码块4");}{System.out.println("访问了父类的实例代码块5");}
public java0608(String name,int age){System.out.println("调用了父类的构造方法6");
}
public class java0609 {public static void main(String[] args) {java0607 t =new java0607("张三",18);
//    t.test();}
}

程序运行完以后我们会发现是415623

也就是秉持静态总体最优先,父类总体优先的规则

实例化第二个对象以后我们也会发现是静态没有第二次调用,与之前很类似

四、protected关键字

package bit.pack;public class demo3protected{protected int a = 8;
}
package bit;
import bit.pack.demo3protected;
public class demo4protected extends demo3protected{public void test() {                //方式一:调用方法来访问protected变量System.out.println(super.a);    //不能在静态方法中访问非静态成员,main方法就加了static}public static void main(String[] args) {demo4protected t = new demo4protected();System.out.println(t.a);        //方式二:实例化子类对象来访问protected变量}
}

这里展示的就是不同包底下的子类访问方式

五、多态

1、两种绑定类型

先一口气定义一个包下三个不同的类,方便展示:

package bit.pack;public class Animal {public int age;public String name;public Animal(int age, String name) {this.age = age;this.name = name;}public void eat(){System.out.println(this.name+"正在吃饭");}
}
package bit.pack;public class Dog2 extends Animal{public Dog2(int age,String name){super(age, name);this.age = age;}public void eat(){System.out.println(this.name+"正在吃狗粮");}public void bark(){System.out.println(this.name+"正在汪汪叫");}}
package bit.pack;public class Test {public static void main(String[] args) {
//        Dog2 dog = new Dog2();
//        Animal animal = dog;    //表示父类的引用指向了子类的对象Animal animal = new Dog2(18,"旺财");  //向上转型animal.eat();}
}

这里的子类和父类的方法用到了重写这个概念

并通过父类的引用指向子类对象的方式实现了向上转型

由于这里引用的是Animal类型,  因此我们发现只能调用Animal下的方法,调用bark方法程序就运行失败了。于是我们得到只能调用父类下方法的结论

然而运行后我们发现结果是子类下的打印:旺财正在吃狗粮


我们把这种现象称为:运行时绑定(动态绑定)

它的发生条件就是:1、发生了向上转型

                                 2、子类重写了父类的方法

                                 3、父类调用了这个方法

那就引申出了另一种绑定:编译时绑定(静态绑定)

简单讲其实就是重载时匹配合适的方法:

 public void class add(int a,int b)public void class add(int a,int b,int c)add(2,3)add(1,2,3)


总结

相关文章:

  • ESP32的spi通讯(Arduino)
  • 高精度模板
  • RAG 系统评估与优化指南:从 RAGAS 到 ARES 的实战应用
  • python: wxpython 4.2 开发一个邮件客户端,能编写邮件,发送邮件及附件
  • Visualized_BGE 安装—多模态嵌入技术
  • 3.1.3_栈的链式存储实现
  • 006__C++类的特殊用法
  • mysql查看表结构语句
  • 【论文阅读34】Attention-ResNet-LSTM(JRMGE2024)
  • HTTP 协议里15种请求方法及示例
  • 《P1878 舞蹈课》
  • 高效I/O处理:模型与多路复用的探讨
  • JDBC基础(1)
  • 快速学习GO语言总结
  • 1.16 Cookie 和 Session
  • Linux进程池详解:从入门到理解
  • 『uniapp』搜索功能+商品列表滚动效果(详细图文注释)
  • 华为OD机试_2025 B卷_数组排列求和(Python,100分)(附详细解题思路)
  • WWDC25中的HDR技术洞察
  • MySQL 锁学习笔记
  • 手机wap网站建设多少钱/如何做网站推广私人
  • 网站26个页面收费/四川餐饮培训学校排名
  • 网站开发哪家公司好/友情链接只有链接
  • seo网站推广seo/网盘资源大全
  • 前端做项目的网站资源/外包网络推广公司怎么选
  • 做网站还用注册商标吗/seo优化文章网站