【JavaSE】解密 继承和多态(下)

逸狼 2024-08-10 13:35:01 阅读 70

前言

紧接着上篇 解密继承和多态(上)~

欢迎关注个人主页:逸狼


创造不易,可以点点赞吗~

如有错误,欢迎指出~



目录

前言

protected关键字

在同一包下同一类可以访问

代码理解

在同一包下不同类可以访问

代码理解

在不同包下的子类可以访问

代码理解

再谈访问权限问题

final关键字

final修饰变量

final修饰类

组合

代码举例

多态

动态绑定

发生条件

向上转型

代码理解

直接赋值

方法传参

作为返回值 

向下转型

父类和子类 方法的重写

要求

注意

@Override

与代码重载的区别

代码理解


protected关键字

接下来我们谈谈protected关键字

在同一包下同一类可以访问

代码理解

如下的代码,用protected修饰c,在同一个包demo1下的同一类Test1下,func方法可以成功访问c

<code>package demo1;

public class Test1 {

public int a;

protected int c=99;

public void func(){

System.out.println(c);

}

}

在同一包下不同类可以访问

在同一个包demo1下面再建一个类TestProtected1

代码理解

调用Test1产生实例化对象test1,可以通过对象test1访问Test1类下的成员c

<code>package demo1;

public class TestProtected1 {

public static void main(String[] args) {

Test1 test1=new Test1();

System.out.println(test1.c);

}

}

在不同包下的子类可以访问

被protected修饰,不管同不同包,只要是子类就都可以 通过super 访问

代码理解

新创建另外一个包demo2,再在下面创建一个类Test2

在Test2类中可以通过super成功访问不同包demo2下Test1类中被protected修饰的c

前提:被继承的类是用public修饰的,在代码中体现就是Test被public修饰了才行

(其中类的权限 只有 两种:一个是用public修饰的,另一个是不用public修饰的)

<code>package demo2;

import demo1.Test1;//要先导包

//继承Test1

public class Test2 extends Test1 {

public void test(){

System.out.println(super.c);

}

public static void main(String[] args) {

// System.out.println(super.c);//会报错,因为main方法中有static修饰,不能用super

}

}

再谈访问权限问题

在上一篇我们讲解了private和public的范围,他们是访问权限的两个极端

private只能在同一包的同一类下访问public是 不管是否同包 不管是否同类 都能被访问

我们可以用下图总结

这里的default不是关键字,表示的是在成员变量前不加任何public、private等关键字

Java不支持多继承

final关键字

final修饰变量

final int SIZE=10;表示SIZE变成常量(不能被修改)

final修饰类

final用于控制继承,被final修饰的类 表示 当前类不可以被继承,此时这个类称为密封类

组合

组合是代码层面的一种写法,是has-a的关系(例如 汽车中的零部件组合成了一辆汽车),仅仅是将一个类的示例作为另一个类的成员变量

代码举例

老师类和学生类组合成了学校类

<code>class Teacher{

}

class Student{

}

class School{

private Teacher[] teachers;

private Student[] students;

}

多态

同一件事不同的对象上产生的效果是不一样

代码理解

当Animal引用的对象不一样(animal1和animal2),调用eat方法,表现出的行为不一样时(Animal1吃狗粮,Animal2吃鸟粮),这就叫做 多态。

class Animal{

public String name;

public int age;

public void eat(){

System.out.println(this.name+" 正在吃~");

}

public Animal(String name, int age) {

this.name = name;

this.age = age;

}

}

class Dog extends Animal{

//Alt键 加上 Enter键快速生成子类构造方法

public Dog(String name, int age) {

super(name, age);

}

@Override//注解 用于帮助检查方法重写是否正确,若重写的方法有问题,他就会报错

public void eat(){

System.out.println(this.name+"正在吃狗粮~");//与父类的eat形成 方法的重写

}

public void bark(){

System.out.println(this.name+" 正在汪汪汪~");

}

}

class Bird extends Animal{

public Bird(String name, int age) {

super(name, age);

}

public void qiqi(){

System.out.println(this.name+" 正在吱吱吱~");

}

public void eat(){

System.out.println(this.name+"正在吃鸟粮~");//与父类的eat形成 方法的重写

}

}

public class Test {

public static void func(Animal animal){

}

public static Animal func2(){

return new Dog("旺财",3);

}

public static void main(String[] args) {

/* Dog dog=new Dog();

Animal animal=dog;//向上转型*/

Animal animal1=new Dog("旺财",3);//向上转型

animal1.eat();

Animal animal2=new Bird("小蜂",1);

animal2.eat();

//animal1.bark();//会报错,因为Animal中没有bark方法,通过父类引用只能访问父类自己的有的

}

}

实现多态的前提是动态绑定~

动态绑定

编译的是父类方法,但是调用的是子类方法

代码理解

编译用的父类Animal中的eat方法,但结果是调用的是子类eat方法

Animal animal1=new Dog("旺财",3);//向上转型

animal1.eat();

 代码结果

发生条件

父类引用子类对象【向上转型】

通过父类引用 调用重写的方法【方法的重写】

向上转型

实际就是创建一个子类对象,将其当成父类对象来使用。

语法格式:父类类型 对象名 = new 子类类型()

优点:让代码实现更简单灵活。缺陷:不能调用到子类特有的方法。

发生向上转型的时机有三种,通过以下代码加以理解

代码理解
直接赋值

<code> public static void main(String[] args) {

/* Dog dog=new Dog();

Animal animal=dog;//向上转型*/

Animal animal1=new Dog("旺财",3);//向上转型

}

方法传参

public static void func(Animal animal){

}

作为返回值 

public static Animal func2(){

return new Dog("旺财",3);

}

向下转型

Animal animal1=new Dog("旺财",3);//向上转型

Dog dog=(Dog)animal1;//向下转型 要将对象animal1强转成Dog类,否则会报错

//因为 不是所有的动物都是狗

父类和子类 方法的重写

重写又叫覆盖、覆写

要求

方法名相同方法的参数列表相同(个数、顺序、类型)方法返回值相同

注意

静态方法 不能 被重写被private修饰的 不能 被重写被final修饰的 不能 被重写如果方法被重写,子类的访问权限要 大于等于 父类的权限

@Override

修饰该方法,说明该方法是重写的

与代码重载的区别

代码理解

<code>

//父类Animal中的eat方法

public void eat(){

System.out.println(this.name+" 正在吃~");

}

//子类Dog中的eat方法

public void eat(){

System.out.println(this.name+"正在吃狗粮~");//与父类的eat形成 方法的重写

}



声明

本文内容仅代表作者观点,或转载于其他网站,本站不以此文作为商业用途
如有涉及侵权,请联系本站进行删除
转载本站原创文章,请注明来源及作者。