【化神系列】 Java继承

简介: JavaSE继承详解

JavaSE继承


一、继承

  • 理解继承
Cat继承了Animal类,其中:Animal类称为父类/基类或超类,Cat可以称为Animal的
子类/派生类,继承之后, 子类可以复用父类中成员,子类在实现时只需关心自己新增加的成员即可

Java中不支持多继承,仅支持A extends B 单继承//A extends B B extends C 多层继承 //A extends C B extends C 不同类继承自同一类。

public class TestExtend {
    public static void main(String[] args) {
       Dog d1 =new Dog();
       d1.name = "xiaocs";
       d1.age = 15;
       Cat c1 = new Cat();
       c1.color = "red";
       c1.type = "波斯";
       c1.friendly();
    }
}
/**
*    Dog未继承animal则无法调用animal中的成员变量和成员方法
*/
class Dog{
    public  int age;

    public String  name;
    
    public void shot(){
        System.out.println("汪汪汪");
    }
}
//cat可以直接调用animal中的成员变量和成员方法
class Cat extends animal{
    public int weight;

    public String sex;
    
    public void eat(){
        System.out.println("嘎嘎吃");
    }

}
class animal{
    public String  color;

    public String type;
    
    public void friendly(){
        System.out.println("对人类友好");
    }

}
  • 子类访问父类

    • 子类和父类不存在同名成员变量

      public class Base {
          int a;
          int b;
      }
      public class Derived extends Base{
          int c;
          public void method(){
              a = 10; // 访问从父类中继承下来的a
              b = 20; // 访问从父类中继承下来的b
              c = 30; // 访问子类自己的c
          }
    • 子类和父类成员变量同名

      public class Base {
          int a;
          int b;
          int c;
      } 
      
      public class Derived extends Base{
          int a; // 与父类中成员a同名,且类型相同
          char b; // 与父类中成员b同名,但类型不同
          public void method(){
              a = 100; 
              b = 101; 
              c = 102; // 子类没有c,访问的肯定是从父类继承下来的c
           // d = 103; // 编译失败,因为父类和子类都没有定义成员变量b
          }
      }
      • 成员变量访问遵循就近原则,自己有优先自己的,与父类同名也优先访问自己的,如果没有则向父类中找。
      • 成员方法亦然,如果子类有则先访问子类的,子类没有先访问父类的,如果存在方法重载,则根据引用的情况,选择合适的方法
  • 在子类方法中,如果想要明确访问父类中成员时,借助super关键字即可

二、super关键字

  • 当父类和子类的成员变量或成员方法相同时,调用父类的成员变量或方法,使用super关键字
class Cat extends animal{
    public int color;

    public String type;

    public void eat(){
    
    //调用父类的成员变量
        super.color = "red";
        super.type = "波斯";
        super.friendly();
        System.out.println("嘎嘎吃");
    }

}
class animal{
    public String  color;
    public String type;

    public void friendly(){
        System.out.println("对人类友好");
    }
}
  • 子类的构造方法(无参数)
  • 若父类显式定义无参或者默认的构造方法,在子类构造方法第一行默认有隐含的super()调用,即调用基类构造方法
public class TestExtend {
    public static void main(String[] args) {
        Cat c = new Cat();
    }
}
class Cat extends animal{
    public int size;
    public String name;

    public Cat(){
        // super(); 
                   //注意子类构造方法中默认会调用基类的无参构造方法:super(), 
                   // 用户没有写时,编译器会自动添加,而且super()必须是子类构造方法中第一条语句,
                   // 并且只能出现一次
        System.out.println("Cat::");
    }
    public void eat(){
        System.out.println("嘎嘎吃");
    }

}
class animal{
    public String  color;
    public String type;

    public animal() {
        System.out.println("animal::");
    }
    public void friendly(){
        System.out.println("对人类友好");
    }
}

image-20220901095550142

子类对象构造时,需要先调用父类构造方法,然后执行子类的构造方法。
  • 子类的构造方法(有参数)
  • 如果父类构造方法是带有参数的,此时需要用户为子类显式定义构造方法,并在子类构造方法中选择合适的父类构造方法调用,否则会编译失败
/**
 * Created with IntelliJ IDEA.
 * Description:
 * User: snutyx
 * Date: 2022-08-30
 * Time: 17:05
 */

public class TestExtend {
    public static void main(String[] args) {
        Cat c = new Cat("red","波斯",45,"xiaos");
    }
}
class Cat extends animal{
    public int size;
    public String name;

    public Cat(String color, String type, int size, String name) {
        super(color, type);
        this.size = size;
        this.name = name;
    }
    public void eat(){
        System.out.println("嘎嘎吃");
    }
}
class animal{
    public String  color;
    public String type;

    public animal(String color, String type) {
        this.color = color;
        this.type = type;
    }
    public void friendly(){
        System.out.println("对人类友好");
    }

}
  • 在子类构造方法中,super(...)调用父类构造时,必须是子类构造函数中第一条语句。
  • super(...)只能在子类构造方法中出现一次,并且不能和this同时出现

✅super和this之间的区别?

  • 相同点:

1.都属于java 中 的关键字

2.只能在类的非静态方法中使用,用来访问 非静态的方法和字段

3.在构造 方法中使用时必须是构造方法当中的第一条语句,并且二者不能同时存在

  • 不同点:

    1.this指的是当前对象的引用,super指的是当前对象对父类部分对象的使用

  1. 在非静态成员方法中,this用来访问本类的方法和属性,super用来访问父类继承下来的方法和属性
  2. 在构造方法当中,this用于调用本类的构造方法,super调用父类的构造方法,两种调用不能同时在构造方法中出现
  3. 构造方法中一定会存在super(...)的调用,用户没有写编译器也会增加,但是this(...)用户不写则没有

三、继承关系下的代码块

public class TestExtend {
    public static void main(String[] args) {
        Cat c = new Cat("red","波斯",45,"xiaos");
    }
}
class Cat extends animal{
    public int size;
    public String name;
    //5.子类的实例代码块
    {
        System.out.println("cat::实例");
    }
    //2.子类的静态代码块
    static{
        System.out.println("cat::static");

    }
    //6.子类的构造方法
    public Cat(String color, String type, int size, String name) {
        super(color, type);
        this.size = size;
        this.name = name;
        System.out.println("cat::构造");
    }
    public void eat(){
        System.out.println("嘎嘎吃");
    }
}
class animal{
    public String  color;
    public String type;

    //3.父类的实例代码块
    {
        System.out.println("animal::实例");
    }
    //1。父类的静态代码块
    static{
        System.out.println("animal::static");

    }
    //4.父类的构造方法
    public animal(String color, String type) {
        this.color = color;
        this.type = type;
        System.out.println("animal::构造");
    }
    public void friendly(){
        System.out.println("对人类友好");
    }

image-20220901103853647

根据执行结果可以的出以下结论:

父类的静态代码块>>子类的静态代码块>>父类的实例代码块>父类的构造方法>>子类的实例代码块>> 子类的构造方法
  • 第二次实例化子类对象时,父类和子类的静态代码块都将不会再执行

四、protected关键字

访问限定:同一个包中的类和不同包中的子类
package TestDemo2;

public class B {
    private int a;
    protected int b;
    public int c;
    int d;
}

在不同包中即package TestDemo 中

image-20220901181200101

只有public和protected修饰的成员可以访问

五、final关键字

  • 修饰变量时: 表示常量即不能修改
final int a = 10; 
a = 20; // 编译出错
  • 修饰类时: 表示此类不能被继承,即不能作为父类
//final修饰
final class animal{
    public String  color;
    public String type;
    public animal(String color, String type) {
        this.color = color;
        this.type = type;
        System.out.println("animal::构造");
    }
}

image-20220901182335720

相关文章
|
5月前
|
Java 程序员
Java中的继承和多态:理解面向对象编程的核心概念
【8月更文挑战第22天】在Java的世界中,继承和多态不仅仅是编程技巧,它们是构建可维护、可扩展软件架构的基石。通过本文,我们将深入探讨这两个概念,并揭示它们如何共同作用于面向对象编程(OOP)的实践之中。你将了解继承如何简化代码重用,以及多态如何为程序提供灵活性和扩展性。让我们启程,探索Java语言中这些强大特性的秘密。
|
2月前
|
Java
在Java中,接口之间可以继承吗?
接口继承是一种重要的机制,它允许一个接口从另一个或多个接口继承方法和常量。
124 1
|
3月前
|
Java 开发者
在Java多线程编程中,创建线程的方法有两种:继承Thread类和实现Runnable接口
【10月更文挑战第20天】在Java多线程编程中,创建线程的方法有两种:继承Thread类和实现Runnable接口。本文揭示了这两种方式的微妙差异和潜在陷阱,帮助你更好地理解和选择适合项目需求的线程创建方式。
40 3
|
3月前
|
Java
在Java多线程编程中,实现Runnable接口通常优于继承Thread类
【10月更文挑战第20天】在Java多线程编程中,实现Runnable接口通常优于继承Thread类。原因包括:1) Java只支持单继承,实现接口不受此限制;2) Runnable接口便于代码复用和线程池管理;3) 分离任务与线程,提高灵活性。因此,实现Runnable接口是更佳选择。
64 2
|
3月前
|
Java
Java中多线程编程的基本概念和创建线程的两种主要方式:继承Thread类和实现Runnable接口
【10月更文挑战第20天】《JAVA多线程深度解析:线程的创建之路》介绍了Java中多线程编程的基本概念和创建线程的两种主要方式:继承Thread类和实现Runnable接口。文章详细讲解了每种方式的实现方法、优缺点及适用场景,帮助读者更好地理解和掌握多线程编程技术,为复杂任务的高效处理奠定基础。
44 2
|
3月前
|
Java 开发者
Java多线程初学者指南:介绍通过继承Thread类与实现Runnable接口两种方式创建线程的方法及其优缺点
【10月更文挑战第20天】Java多线程初学者指南:介绍通过继承Thread类与实现Runnable接口两种方式创建线程的方法及其优缺点,重点解析为何实现Runnable接口更具灵活性、资源共享及易于管理的优势。
49 1
|
4月前
|
Java 编译器
封装,继承,多态【Java面向对象知识回顾①】
本文回顾了Java面向对象编程的三大特性:封装、继承和多态。封装通过将数据和方法结合在类中并隐藏实现细节来保护对象状态,继承允许新类扩展现有类的功能,而多态则允许对象在不同情况下表现出不同的行为,这些特性共同提高了代码的复用性、扩展性和灵活性。
封装,继承,多态【Java面向对象知识回顾①】
|
3月前
|
Java 测试技术 编译器
Java零基础-继承详解!
【10月更文挑战第4天】Java零基础教学篇,手把手实践教学!
52 2
|
3月前
|
Java 编译器
在Java中,关于final、static关键字与方法的重写和继承【易错点】
在Java中,关于final、static关键字与方法的重写和继承【易错点】
36 5
|
3月前
|
Java
java继承和多态详解
java继承和多态详解
54 5