主页
  • 主页
  • 分类
  • 热文
  • 教程
  • 面试
  • 标签
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 抽象类

含有 abstract 关键字声明的 Java 类称为抽象类。Java 抽象类可以包含或者不包含抽象方法(即没有方法体的方法)。但是,如果一个类至少有一个抽象方法,则该类必须声明为抽象类。抽象类不能被实例化,要使用抽象类,你需要继承它并在子类中提供抽象方法的实现。

抽象类示例

以下示例展示了如何在 Java 中创建一个抽象类。只需在类声明前加上 abstract 关键字即可创建一个抽象类。

// 文件名:Employee.java
public abstract class Employee {
    private String name;
    private String address;
    private int number;

    public Employee(String name, String address, int number) {
        System.out.println("构造 Employee");
        this.name = name;
        this.address = address;
        this.number = number;
    }
    
    public double computePay() {
        System.out.println("在 Employee computePay 内部");
        return 0.0;
    }
    
    public void mailCheck() {
        System.out.println("邮寄支票到 " + this.name + " " + this.address);
    }

    public String toString() {
        return name + " " + address + " " + number;
    }

    public String getName() {
        return name;
    }
  
    public String getAddress() {
        return address;
    }
    
    public void setAddress(String newAddress) {
        address = newAddress;
    }
  
    public int getNumber() {
        return number;
    }
}

除了抽象方法外,Employee 类与普通的 Java 类并无二致。该类现在是抽象的,但它仍然拥有三个字段、七个方法和一个构造函数。

尝试以下方式实例化 Employee 类:

// 文件名:AbstractDemo.java
public class AbstractDemo {

    public static void main(String[] args) {
        // 下面的代码会导致错误
        Employee e = new Employee("George W.", "Houston, TX", 43);
        System.out.println("\n 使用 Employee 引用调用 mailCheck --");
        e.mailCheck();
    }
}

当你编译上述类时,会得到如下错误:

Employee.java:46: Employee 是抽象的;不能实例化
      Employee e = new Employee("George W.", "Houston, TX", 43);
                   ^
1 个错误

继承抽象类

我们可以像继承具体类一样继承 Employee 类的属性,如下所示:

// 文件名:Salary.java
public class Salary extends Employee {
    private double salary;   // 年薪

    public Salary(String name, String address, int number, double salary) {
        super(name, address, number);
        setSalary(salary);
    }
    
    public void mailCheck() {
        System.out.println("在 Salary 类的 mailCheck 内部 ");
        System.out.println("邮寄支票到 " + getName() + " 工资 " + salary);
    }
  
    public double getSalary() {
        return salary;
    }
    
    public void setSalary(double newSalary) {
        if(newSalary >= 0.0) {
            salary = newSalary;
        }
    }
    
    public double computePay() {
        System.out.println("计算 " + getName() + " 的工资");
        return salary / 52;
    }
}

在这里,你不能实例化 Employee 类,但你可以实例化 Salary 类,并使用这个实例访问 Employee 类的所有三个字段和七个方法,如下所示:

// 文件名:AbstractDemo.java
public class AbstractDemo {

    public static void main(String[] args) {
        Salary s = new Salary("Mohd Mohtashim", "Ambehta, UP", 3, 3600.00);
        Employee e = new Salary("John Adams", "Boston, MA", 2, 2400.00);
        System.out.println("使用 Salary 引用调用 mailCheck --");
        s.mailCheck();
        System.out.println("\n 使用 Employee 引用调用 mailCheck --");
        e.mailCheck();
    }
}

输出:

构造 Employee
构造 Employee
使用 Salary 引用调用 mailCheck --
在 Salary 类的 mailCheck 内部
邮寄支票到 Mohd Mohtashim 工资 3600.0

使用 Employee 引用调用 mailCheck --
在 Salary 类的 mailCheck 内部
邮寄支票到 John Adams 工资 2400.0

Java 抽象方法

如果你想让一个类包含某个特定的方法,但实际的实现由子类来决定,那么可以在父类中声明该方法为抽象方法。使用 abstract 关键字声明抽象方法。在方法声明中,抽象方法需要在方法名前放置 abstract 关键字,并且不需要方法体;而是以分号(;)结束而不是花括号({})。

以上就是关于 Java 中抽象类和抽象方法的基础介绍和示例。

Java 中实现抽象方法的示例

下面是一个实现抽象方法的例子:

// 文件名:Employee.java
public abstract class Employee {
    private String name;
    private String address;
    private int number;
    
    public abstract double computePay();

    public Employee(String name, String address, int number) {
        System.out.println("构造 Employee");
        this.name = name;
        this.address = address;
        this.number = number;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public String getAddress() {
        return address;
    }

    public void setAddress(String address) {
        this.address = address;
    }

    public int getNumber() {
        return number;
    }

    public void setNumber(int number) {
        this.number = number;
    }
}

将一个方法声明为抽象的有两个后果:

  • 包含它的类必须声明为抽象类。
  • 任何继承当前类的类都必须要么覆盖抽象方法,要么声明自己为抽象类。

注意,最终,后代类必须实现抽象方法;否则,就会有一个无法实例化的抽象类层次结构。

假设 Salary 类继承了 Employee 类,那么它应该实现 computePay() 方法,如下所示:

// 文件名:Salary.java
public class Salary extends Employee {
    private double salary;   // 年薪

    public Salary(String name, String address, int number, double salary) {
        super(name, address, number);
        this.salary = salary;
    }

    @Override
    public double computePay() {
        System.out.println("计算 " + getName() + " 的薪水支付");
        return salary / 52;
    }
}

接下来,我们可以通过实例化 Salary 类并调用 computePay() 方法来测试我们的代码:

// 文件名:AbstractDemo.java
public class AbstractDemo {
    public static void main(String[] args) {
        Salary s = new Salary("Mohd Mohtashim", "Ambehta, UP", 3, 3600.00);
        System.out.println("薪水: " + s.computePay());
    }
}

输出将是:

构造 Employee
计算 Mohd Mohtashim 的薪水支付
薪水: 69.23076923076923

此输出表明 Employee 类被正确地继承,并且 Salary 类实现了 computePay() 方法。当创建 Salary 类的实例并调用 computePay() 时,它返回了每周的薪水数额。

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

关注阅读号

联系二维码

联系我们

© 2024 Yoagoa. All rights reserved.

粤ICP备18007391号

站点地图