主页
  • 主页
  • 分类
  • 热文
  • 教程
  • 面试
  • 标签
Java

Java 基础

Java 主页
Java 概述
Java 历史
Java 功能
Java 与 C++
Java JVM(Java虚拟机)
Java JDK、JRE 和 JVM
Java Hello World 程序
Java 环境设置
Java 基本语法
Java 变量类型
Java 数据类型
Java 类型转换
Java Unicode 系统
Java 基本运算符
Java 注释
Java 用户输入
Java 日期和时间

Java 控制语句

Java 循环控制
Java 决策结构
Java if-else 语句
Java switch 语句
Java for 循环
Java for each 循环
Java while 循环
Java do...while 循环
Java break 语句
Java continue 语句

Java 面向对象编程

Java OOP概念
Java 类和对象
Java 类属性
Java 类方法
Java 方法
Java 变量作用域
Java 构造函数
Java 访问修饰符
Java 继承
Java 聚合
Java 多态
Java 覆盖
Java 方法重载
Java 动态绑定
Java 静态绑定
Java 实例初始化块
Java 抽象
Java 封装
Java 接口
Java 包
Java 内部类
Java 静态类
Java 匿名类
Java 单例类
Java 包装类
Java 枚举类
Java 枚举构造函数
Java 枚举字符串

Java 内置类

Java 数字
Java 布尔值
Java 字符
Java 数组
Java 数学类

Java 文件处理

Java 文件
Java 创建文件
Java 写入文件
Java 读取文件
Java 删除文件
Java 目录操作
Java I/O流

Java 错误和异常

Java 异常
Java Try Catch
Java try-with-resources
Java 多个 Catch
Java 嵌套 try
Java finally
Java 抛出异常
Java 异常传播
Java 内置异常
Java 自定义异常

Java 多线程

Java 多线程
Java 线程生命周期
Java 创建线程
Java 启动线程
Java 加入线程
Java 命名线程
Java 线程调度
Java 线程池
Java 主线程
Java 线程优先级
Java 守护线程
Java 线程组
Java JVM 关闭

Java 同步

Java 线程同步
Java 块同步
Java 静态同步
Java 线程间通信
Java 线程死锁
Java 中断线程
Java 线程控制
Java 可重入锁

Java 网络

Java 网络编程
Java 套接字编程
Java URL 处理
Java URL 类
Java URLConnection 类
Java HttpURLConnection 类
Java Socket 类
Java 泛型

Java 集合

Java 集合框架
Java 集合接口

Java 接口

Java 列表接口
Java 队列接口
Java 映射接口
Java SortedMap 接口
Java 集合(Set)接口
Java SortedSet 接口

Java 数据结构

Java 数据结构
Java 枚举接口

Java 集合算法

Java 迭代器
Java 比较器
Java Comparable 接口

Java 高级

Java 命令行参数
Java Lambda 表达式
Java 发送电子邮件
Java 小应用程序
Java Javadoc
Java 自动装箱和拆箱
Java mismatch() 方法
Java REPL
Java 多版本发布 JAR
Java 私有接口方法
Java 金刚石操作符
Java 多分辨率图像 API
Java 集合的工厂方法
Java 模块系统
Java Nashorn 引擎
Java Optional 类
Java 方法引用
Java 功能接口
Java 默认方法
Java Base64 工具类
Java Switch 表达式
Java Collectors.teeing() 方法
Java 基准测试
Java 文本块
Java 动态CDS
Java ZGC
Java NullPointerException
Java jpackage
Java 密封类
Java 记录
Java 隐藏类
Java instanceof
Java 紧凑数字格式化
Java 垃圾回收
Java JIT 编译器

Java 杂项

Java 递归
Java 正则表达式
Java 序列化
Java 字符串类
Java 进程 API
Java Stream API
Java @Deprecated 注释
Java CompletableFuture API
Java Streams
Java 日期时间 API

基础

Java 主页
Java 概述
Java 历史
Java 功能
Java 与 C++
Java JVM(Java虚拟机)
Java JDK、JRE 和 JVM
Java Hello World 程序
Java 环境设置
Java 基本语法
Java 变量类型
Java 数据类型
Java 类型转换
Java Unicode 系统
Java 基本运算符
Java 注释
Java 用户输入
Java 日期和时间

控制语句

Java 循环控制
Java 决策结构
Java if-else 语句
Java switch 语句
Java for 循环
Java for each 循环
Java while 循环
Java do...while 循环
Java break 语句
Java continue 语句

面向对象编程

Java OOP概念
Java 类和对象
Java 类属性
Java 类方法
Java 方法
Java 变量作用域
Java 构造函数
Java 访问修饰符
Java 继承
Java 聚合
Java 多态
Java 覆盖
Java 方法重载
Java 动态绑定
Java 静态绑定
Java 实例初始化块
Java 抽象
Java 封装
Java 接口
Java 包
Java 内部类
Java 静态类
Java 匿名类
Java 单例类
Java 包装类
Java 枚举类
Java 枚举构造函数
Java 枚举字符串

内置类

Java 数字
Java 布尔值
Java 字符
Java 数组
Java 数学类

文件处理

Java 文件
Java 创建文件
Java 写入文件
Java 读取文件
Java 删除文件
Java 目录操作
Java I/O流

错误和异常

Java 异常
Java Try Catch
Java try-with-resources
Java 多个 Catch
Java 嵌套 try
Java finally
Java 抛出异常
Java 异常传播
Java 内置异常
Java 自定义异常

多线程

Java 多线程
Java 线程生命周期
Java 创建线程
Java 启动线程
Java 加入线程
Java 命名线程
Java 线程调度
Java 线程池
Java 主线程
Java 线程优先级
Java 守护线程
Java 线程组
Java JVM 关闭

同步

Java 线程同步
Java 块同步
Java 静态同步
Java 线程间通信
Java 线程死锁
Java 中断线程
Java 线程控制
Java 可重入锁

网络

Java 网络编程
Java 套接字编程
Java URL 处理
Java URL 类
Java URLConnection 类
Java HttpURLConnection 类
Java Socket 类
Java 泛型

集合

Java 集合框架
Java 集合接口

接口

Java 列表接口
Java 队列接口
Java 映射接口
Java SortedMap 接口
Java 集合(Set)接口
Java SortedSet 接口

数据结构

Java 数据结构
Java 枚举接口

集合算法

Java 迭代器
Java 比较器
Java Comparable 接口

高级

Java 命令行参数
Java Lambda 表达式
Java 发送电子邮件
Java 小应用程序
Java Javadoc
Java 自动装箱和拆箱
Java mismatch() 方法
Java REPL
Java 多版本发布 JAR
Java 私有接口方法
Java 金刚石操作符
Java 多分辨率图像 API
Java 集合的工厂方法
Java 模块系统
Java Nashorn 引擎
Java Optional 类
Java 方法引用
Java 功能接口
Java 默认方法
Java Base64 工具类
Java Switch 表达式
Java Collectors.teeing() 方法
Java 基准测试
Java 文本块
Java 动态CDS
Java ZGC
Java NullPointerException
Java jpackage
Java 密封类
Java 记录
Java 隐藏类
Java instanceof
Java 紧凑数字格式化
Java 垃圾回收
Java JIT 编译器

杂项

Java 递归
Java 正则表达式
Java 序列化
Java 字符串类
Java 进程 API
Java Stream API
Java @Deprecated 注释
Java CompletableFuture API
Java Streams
Java 日期时间 API

Java 继承


上一章 下一章

Java 继承是 Java 面向对象编程(OOP)的一个重要概念。继承是一个过程,其中一个类(子类)获得另一个类(父类)的属性(方法和字段)。通过使用继承,信息在一个层次结构中变得易于管理。

继承中的类可以分为两类:

  • 子类(Subclass 或 Derived class 或 Child class):继承其他类的属性的类。
  • 父类(Superclass 或 Base class 或 Parent class):其属性被继承的类。

Java 继承的必要性

  • 代码重用:继承的基本需求是为了重用特性。如果已经定义了一些功能,那么可以轻松地在其他类和包中使用它们。
  • 扩展性:继承有助于扩展类的功能。如果有一个带有某些功能的基础类,那么可以通过继承在派生类中扩展它们。
  • 方法重写:继承是实现多态性的概念之一,即方法重写。
  • 实现抽象:另一个面向对象的概念——抽象也需要继承。

Java 继承的实现

在 Java 中实现(使用)继承使用 extends 关键字。这会将基类的属性(属性或/和方法)继承到派生类。extends 这个词意味着扩展功能,即特征的可扩展性。

继承语法

class Super {
   .....
   .....
}
class Sub extends Super {
   .....
   .....
}

Java 继承示例

以下示例展示了 Java 继承。在这个示例中,你可以看到两个类:Calculation 和 My_Calculation。

使用 extends 关键字,My_Calculation 继承了 Calculation 类的方法 addition() 和 Subtraction()。

将以下程序复制粘贴到名为 My_Calculation.java 的文件中:

class Calculation {
   int z;
 
   public void addition(int x, int y) {
      z = x + y;
      System.out.println("The sum of the given numbers:" + z);
   }
 
   public void Subtraction(int x, int y) {
      z = x - y;
      System.out.println("The difference between the given numbers:" + z);
   }
}

public class My_Calculation extends Calculation {
   public void multiplication(int x, int y) {
      z = x * y;
      System.out.println("The product of the given numbers:" + z);
   }
 
   public static void main(String args[]) {
      int a = 20, b = 10;
      My_Calculation demo = new My_Calculation();
      demo.addition(a, b);
      demo.Subtraction(a, b);
      demo.multiplication(a, b);
   }
}

编译并执行上述代码:

javac My_Calculation.java
java My_Calculation

执行程序后,会产生以下结果:

The sum of the given numbers:30
The difference between the given numbers:10
The product of the given numbers:200

在这个程序中,当创建 My_Calculation 类的对象时,基类的内容会被复制到派生类中。这就是为什么可以使用派生类的对象来访问基类的成员。

遗产

继承

父类的引用变量可以持有派生类的对象,但是使用该变量只能访问父类的成员,所以为了访问两个类的成员,建议总是创建指向派生类的引用变量。

如果你考虑上面的程序,可以像下面这样实例化类。但是使用父类的引用变量(如这里的 cal),你不能调用属于派生类 My_Calculation 的方法 multiplication()。

Calculation demo = new My_Calculation();
demo.addition(a, b);
demo.Subtraction(a, b);

注意:子类继承了父类的所有成员(字段、方法和嵌套类)。构造器不是成员,所以它们不会被子类继承,但是可以通过子类调用父类的构造器。

Java 继承:super 关键字

super 关键字与 this 关键字类似。以下是 super 关键字使用的场景:

  • 用于区分同名的父类和子类成员。
  • 用于从子类中调用父类的构造器。

区分成员

如果一个类继承了另一个类的属性,并且父类成员与子类成员具有相同的名称,我们可以使用 super 关键字来区分这些变量。

super.variable
super.method();

示例代码

下面的程序展示了 super 关键字的用法。

在给定的程序中,有两个类 Sub_class 和 Super_class,它们都有一个名为 display() 的方法,但实现不同,并且都有一个名为 num 的变量,但值不同。我们在调用两个类的 display() 方法,并打印两个类中 num 变量的值。这里可以看到我们使用 super 关键字来区分父类和子类的成员。

将此程序复制粘贴到名为 Sub_class.java 的文件中。

class Super_class {
   int num = 20;

   // 父类的 display 方法
   public void display() {
      System.out.println("This is the display method of superclass");
   }
}

public class Sub_class extends Super_class {
   int num = 10;

   // 子类的 display 方法
   public void display() {
      System.out.println("This is the display method of subclass");
   }

   public void my_method() {
      // 实例化子类
      Sub_class sub = new Sub_class();

      // 调用子类的 display() 方法
      sub.display();

      // 调用父类的 display() 方法
      super.display();

      // 打印子类中名为 num 的变量的值
      System.out.println("value of the variable named num in sub class:" + sub.num);

      // 打印父类中名为 num 的变量的值
      System.out.println("value of the variable named num in super class:" + super.num);
   }

   public static void main(String args[]) {
      Sub_class obj = new Sub_class();
      obj.my_method();
   }
}

编译并执行上述代码:

javac Sub_class.java
java Sub_class

执行程序后,你会得到以下结果:

This is the display method of subclass
This is the display method of superclass
value of the variable named num in sub class:10
value of the variable named num in super class:20

调用父类构造器

如果一个类继承了另一个类的属性,子类会自动获得父类的默认构造器。但如果想调用父类的带参数构造器,则需要使用 super 关键字,如下所示:

super(values);

示例代码

下面的程序展示了如何使用 super 关键字来调用父类的带参数构造器。这个程序包含一个父类和一个子类,其中父类包含一个接受整数值的带参数构造器,我们使用 super 关键字来调用父类的带参数构造器。

将以下程序复制粘贴到名为 Subclass.java 的文件中:

class Superclass {
   int age;

   Superclass(int age) {
      this.age = age;    
   }

   public void getAge() {
      System.out.println("The value of the variable named age in super class is: " + age);
   }
}

public class Subclass extends Superclass {
   Subclass(int age) {
      super(age);
   }

   public static void main(String args[]) {
      Subclass s = new Subclass(24);
      s.getAge();
   }
}

编译并执行上述代码:

javac Subclass.java
java Subclass

输出:

The value of the variable named age in super class is: 24

IS-A 关系

IS-A 是一种表示方式:这个对象是那个对象的一种类型。让我们看看 extends 关键字是如何实现继承的。

public class Animal {
}

public class Mammal extends Animal {
}

public class Reptile extends Animal {
}

public class Dog extends Mammal {
}

根据上面的例子,从面向对象的角度来看,以下陈述是正确的:

  • 动物(Animal)是哺乳动物(Mammal)类的超类。
  • 动物(Animal)是爬行动物(Reptile)类的超类。
  • 哺乳动物(Mammal)和爬行动物(Reptile)是动物(Animal)类的子类。
  • 狗(Dog)既是哺乳动物(Mammal)也是动物(Animal)类的子类。

现在,如果考虑到 IS-A 关系,我们可以这么说:

  • 哺乳动物(Mammal)是一种动物(Animal)。
  • 爬行动物(Reptile)是一种动物(Animal)。
  • 狗(Dog)是一种哺乳动物(Mammal)。
  • 因此:狗(Dog)也是一种动物(Animal)。

通过使用 instanceof 操作符,我们可以确认哺乳动物确实是动物的一种。

示例

class Animal {
}

class Mammal extends Animal {
}

class Reptile extends Animal {
}

public class Dog extends Mammal {

   public static void main(String args[]) {
      Animal a = new Animal();
      Mammal m = new Mammal();
      Dog d = new Dog();

      System.out.println(m instanceof Animal);
      System.out.println(d instanceof Mammal);
      System.out.println(d instanceof Animal);
   }
}

输出:

true
true
true

既然我们已经了解了 extends 关键字,让我们看看 implements 关键字是如何用于实现 IS-A 关系的。

通常情况下,implements 关键字用于让类继承接口的属性。类永远不能扩展接口。

示例

public interface Animal {
}

public class Mammal implements Animal {
}

public class Dog extends Mammal {
}

请注意,在实际的类设计中,implements 关键字是用来实现接口的,而不是继承。接口中的方法必须在实现类中提供具体的实现。

Java 继承:instanceof 关键字

instanceof 关键字用于确定一个对象是否属于特定的类或实现了某个接口。这是一个非常有用的工具,可以帮助确定运行时对象的确切类型。下面的示例展示了如何使用 instanceof 关键字来检查 Mammal 是否实际上是一个 Animal,以及 Dog 是否实际上是一个 Animal。

interface Animal {}
class Mammal implements Animal {}

public class Dog extends Mammal {
    public static void main(String[] args) {
        Mammal m = new Mammal();
        Dog d = new Dog();

        System.out.println(m instanceof Animal); // 输出 true
        System.out.println(d instanceof Mammal); // 输出 true
        System.out.println(d instanceof Animal); // 输出 true
    }
}

HAS-A 关系

HAS-A 关系主要是基于类的使用情况。它决定了一个类是否拥有另一个类的对象。这种关系有助于减少代码重复以及潜在的错误。

下面的示例展示了 Van 类拥有了 Speed 类的一个实例,这意味着 Van 类有一个 Speed 属性。

public class Vehicle {}
public class Speed {}
public class Van extends Vehicle {
    private Speed sp;
}

通过将速度相关的逻辑封装在单独的 Speed 类中,我们不需要将所有与速度有关的代码都放在 Van 类内,这使得 Speed 类可以在多个应用中重用。

Java 继承的类型

在 Java 中,主要有三种类型的继承:单一继承、多级继承和层次继承。Java 不支持多重继承或多面继承。

继承类型

1. 单一继承(Single Inheritance)

单一继承是指只有一个基类和一个派生类的继承。单一(或单层)继承只从一个基类继承数据到一个派生类。

class One {
    public void printOne() {
        System.out.println("printOne() method of One class.");
    }
}

public class Main extends One {
    public static void main(String[] args) {
        Main obj = new Main();
        obj.printOne(); // 输出 "printOne() method of One class."
    }
}

2. 多级继承(Multilevel Inheritance)

多级继承是指一个基类被派生类继承,而这个派生类又被另一个派生类继承的情况。多级继承涉及到多个基类。

class One {
    public void printOne() {
        System.out.println("printOne() method of One class.");
    }
}

class Two extends One {
    public void printTwo() {
        System.out.println("printTwo() method of Two class.");
    }
}

public class Main extends Two {
    public static void main(String[] args) {
        Main obj = new Main();
        obj.printOne(); // 输出 "printOne() method of One class."
        obj.printTwo(); // 输出 "printTwo() method of Two class."
    }
}

3. 层次继承(Hierarchical Inheritance)

层次继承是指只有一个基类但有多个派生类的情况。

class One {
    public void printOne() {
        System.out.println("printOne() Method of Class One");
    }
}

class Two extends One {
    public void printTwo() {
        System.out.println("Two() Method of Class Two");
    }
}

class Three extends One {
    public void printThree() {
        System.out.println("printThree() Method of Class Three");
    }
}

public class Main {
    public static void main(String[] args) {
        Two obj1 = new Two();
        Three obj2 = new Three();

        obj1.printOne(); // 输出 "printOne() Method of Class One"
        obj2.printOne(); // 输出 "printOne() Method of Class One"
    }
}

注意事项

尽管 Java 不支持多重继承(一个类继承自多个基类),但它允许一个类实现一个或多个接口。这种方式帮助 Java 解决了多重继承的问题。例如:

public class MyClass implements InterfaceOne, InterfaceTwo {
    // 实现接口中的方法
}

在这种情况下,MyClass 可以同时实现 InterfaceOne 和 InterfaceTwo 接口中声明的方法。这种方式使得 Java 能够避免多重继承带来的问题,同时仍然允许一定程度上的“多重行为”通过接口实现。

上一章 下一章
阅读号二维码

关注阅读号

联系二维码

联系我们

© 2024 Yoagoa. All rights reserved.

粤ICP备18007391号

站点地图