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

前言

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

欢迎关注个人主页:逸狼

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

如有错误,欢迎指出~

目录

前言

protected关键字

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

代码理解

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

代码理解

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

代码理解

再谈访问权限问题

final关键字

final修饰变量

final修饰类

组合

代码举例

多态

动态绑定

发生条件

向上转型

代码理解

直接赋值

方法传参

作为返回值 

向下转型

父类和子类 方法的重写

要求

注意

@Override

与代码重载的区别

代码理解


protected关键字

接下来我们谈谈protected关键字

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

代码理解

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

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

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修饰的)

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的关系(例如 汽车中的零部件组合成了一辆汽车),仅仅是将一个类的示例作为另一个类的成员变量

代码举例

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

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 子类类型()

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

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

代码理解
直接赋值
    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

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

与代码重载的区别

代码理解
    
//父类Animal中的eat方法
    public void eat(){
        System.out.println(this.name+" 正在吃~");
    }

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

版权声明:本文为博主作者:逸狼原创文章,版权归属原作者,如果侵权,请联系我们删除!

原文链接:https://blog.csdn.net/2301_80898480/article/details/137274696

共计人评分,平均

到目前为止还没有投票!成为第一位评论此文章。

(0)
社会演员多的头像社会演员多普通用户
上一篇 2024年4月10日
下一篇 2024年4月10日

相关推荐