Java - 抽象
根据字典,抽象 是处理想法而不是事件的质量。例如,当您考虑电子邮件的情况时,复杂的细节(例如您发送电子邮件后会发生什么),您的电子邮件服务器使用的协议对用户是隐藏的。因此,要发送电子邮件,您只需输入内容,提及收件人地址,然后单击发送即可。
同样在面向对象的编程中,抽象是一个向用户隐藏实现细节的过程,只有功能将提供给用户。换句话说,用户将获得关于对象做什么而不是它如何做的信息。
在Java中,抽象是通过抽象类和接口实现的。
抽象类
一个包含 abstract 的类 其声明中的关键字称为抽象类。
-
抽象类可能包含也可能不包含抽象方法 ,即没有主体的方法( public void get(); )
-
但是,如果一个类至少有一个抽象方法,那么这个类必须 被声明为抽象的。
-
如果一个类被声明为抽象的,它就不能被实例化。
-
要使用一个抽象类,你必须从另一个类继承它,并为其中的抽象方法提供实现。
-
如果你继承一个抽象类,你必须提供其中所有抽象方法的实现。
示例
本节为您提供抽象类的示例。要创建抽象类,只需使用 abstract 类声明中类关键字之前的关键字。
/* File name : 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("Constructing an Employee"); this.name = name; this.address = address; this.number = number; } public double computePay() { System.out.println("Inside Employee computePay"); return 0.0; } public void mailCheck() { System.out.println("Mailing a check to " + 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 类 -
/* File name : AbstractDemo.java */ public class AbstractDemo { public static void main(String [] args) { /* Following is not allowed and would raise error */ Employee e = new Employee("George W.", "Houston, TX", 43); System.out.println("\n Call mailCheck using Employee reference--"); e.mailCheck(); } }
当你编译上面的类时,它会给你以下错误 -
Employee.java:46: Employee is abstract; cannot be instantiated Employee e = new Employee("George W.", "Houston, TX", 43); ^ 1 error
继承抽象类
我们可以通过以下方式继承 Employee 类的属性,就像具体类一样 -
示例
/* File name : Salary.java */ public class Salary extends Employee { private double salary; // Annual salary public Salary(String name, String address, int number, double salary) { super(name, address, number); setSalary(salary); } public void mailCheck() { System.out.println("Within mailCheck of Salary class "); System.out.println("Mailing check to " + getName() + " with salary " + salary); } public double getSalary() { return salary; } public void setSalary(double newSalary) { if(newSalary >= 0.0) { salary = newSalary; } } public double computePay() { System.out.println("Computing salary pay for " + getName()); return salary/52; } }
这里不能实例化Employee类,但可以实例化Salary类,使用这个实例可以访问Employee类的全部三个字段和七个方法,如下所示。
/* File name : 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("Call mailCheck using Salary reference --"); s.mailCheck(); System.out.println("\n Call mailCheck using Employee reference--"); e.mailCheck(); } }
这会产生以下结果 -
输出
Constructing an Employee Constructing an Employee Call mailCheck using Salary reference -- Within mailCheck of Salary class Mailing check to Mohd Mohtashim with salary 3600.0 Call mailCheck using Employee reference-- Within mailCheck of Salary class Mailing check to John Adams with salary 2400.0
抽象方法
如果你想让一个类包含一个特定的方法,但又想让该方法的实际实现由子类决定,你可以在父类中将方法声明为抽象。
-
抽象 关键字用于将方法声明为抽象。
-
你必须把 abstract 方法声明中方法名前的关键字。
-
抽象方法包含方法签名,但没有方法体。
-
抽象方法不是花括号,而是在末尾有一个符号冒号 (;)。
下面是一个抽象方法的例子。
示例
public abstract class Employee { private String name; private String address; private int number; public abstract double computePay(); // Remainder of class definition }
将方法声明为抽象有两个后果 -
-
包含它的类必须声明为抽象。
-
任何继承当前类的类都必须重写抽象方法或将自己声明为抽象。
注意 − 最终,子类必须实现抽象方法;否则,您将拥有无法实例化的抽象类层次结构。
假设 Salary 类继承了 Employee 类,那么它应该实现 computePay() 方法如下图-
/* File name : Salary.java */ public class Salary extends Employee { private double salary; // Annual salary public double computePay() { System.out.println("Computing salary pay for " + getName()); return salary/52; } // Remainder of class definition }
java