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