抽象類定義
- 在面向?qū)ο蟮母拍钪?,所有的對象都是通過類來描繪的,但是反過來,并不是所有的類都是用來描繪對象的,如果一個(gè)類中沒有包含足夠的信息來描繪一個(gè)具體的對象,這樣的類就是抽象類。
- 抽象類除了不能實(shí)例化對象之外,類的其它功能依然存在,成員變量、成員方法和構(gòu)造方法的訪問方式和普通類一樣。
- 由于抽象類不能實(shí)例化對象,所以抽象類必須被繼承,才能被使用。也是因?yàn)檫@個(gè)原因,通常在設(shè)計(jì)階段決定要不要設(shè)計(jì)抽象類。
- 父類包含了子類集合的常見的方法,但是由于父類本身是抽象的,所以不能使用這些方法。
- 在Java中抽象類表示的是一種繼承關(guān)系,一個(gè)類只能繼承一個(gè)抽象類,而一個(gè)類卻可以實(shí)現(xiàn)多個(gè)接口。
多態(tài)抽象復(fù)雜例子
Employee(父類超類)
package finance;
/*
* 父類(超類)
* 抽象類 不能實(shí)例化,但是能被繼承
* 構(gòu)造函數(shù)里 this.name 是成員變量private String name,name 為參數(shù)String name;
* */
public abstract class Employee {
private String name;
private String address;
private int number;
// 父類(超類)構(gòu)造函數(shù)
public Employee(String name, String address, int number) {
System.out.println("Constructing an Employee");
//構(gòu)造函數(shù)里 this.name 是成員變量private String name,name 為參數(shù)String name;
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("郵寄支票給:" + 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;
}
}
Salary(子類繼承父類)
package finance;
public class Salary extends Employee {
private double salary;//全年工資
// 繼承父類,子類構(gòu)造函數(shù)
public Salary(String name,String address,int number,double salary){
// 傳參繼承的父類Employee
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("計(jì)算公司,付給:" + getName());
return salary/52;
}
}
輸出
(1)報(bào)錯(cuò) 抽象類不能實(shí)例化
package finance;
public class AbstractDemo {
public static void main(String[] args) {
// TODO Auto-generated method stub
/* 以下是不允許的,會(huì)引發(fā)錯(cuò)誤 */
Employee e = new Employee("George W.", "Houston, TX", 43);
System.out.println("\n Call mailCheck using Employee reference--");
e.mailCheck();
}
}
圖片
(2)成功
package finance;
public class VirtualDemo {
public static void main(String[] args) {
// TODO Auto-generated method stub
Salary s = new Salary("員工 A", "北京", 3 , 3600.00);
Employee e = new Salary("員工 B", "上海", 2, 2400.00);
System.out.println("使用 Salary 的引用調(diào)用 mailCheck -- ");
s.mailCheck();
System.out.println("\n使用 Employee 的引用調(diào)用 mailCheck--");
e.mailCheck();
}
}
--print--
Constructing an Employee
Constructing an Employee
使用 Salary 的引用調(diào)用 mailCheck --
Salary 類的 mailCheck 方法
郵寄支票給:員工 A,工資為:3600.0
使用 Employee 的引用調(diào)用 mailCheck--
Salary 類的 mailCheck 方法
郵寄支票給:員工 B,工資為:2400.0
抽象方法
1.如果你想設(shè)計(jì)這樣一個(gè)類,該類包含一個(gè)特別的成員方法,該方法的具體實(shí)現(xiàn)由它的子類確定,那么你可以在父類中聲明該方法為抽象方法。
2.Abstract關(guān)鍵字同樣可以用來聲明抽象方法,抽象方法只包含一個(gè)方法名,而沒有方法體。
3.抽象方法沒有定義,方法名后面直接跟一個(gè)分號(hào),而不是花括號(hào)。
public abstract class Employee {
private String name;
private String address;
private int number;
public abstract double computePay();
//其余代碼
}
聲明抽象方法會(huì)造成以下兩個(gè)結(jié)果:
- 如果一個(gè)類包含抽象方法,那么該類必須是抽象類。
- 任何子類必須重寫父類的抽象方法,或者聲明自身為抽象類。
繼承抽象方法的子類必須重寫該方法。否則,該子類也必須聲明為抽象類。最終,必須有子類實(shí)現(xiàn)該抽象方法,否則,從最初的父類到最終的子類都不能用來實(shí)例化對象。
如果Salary類繼承了Employee類,那么它必須實(shí)現(xiàn)computePay()方法:
/* 文件名 : 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;
}
//其余代碼
}
抽象類總結(jié)規(guī)定
- 抽象類不能被實(shí)例化(初學(xué)者很容易犯的錯(cuò)),如果被實(shí)例化,就會(huì)報(bào)錯(cuò),編譯無法通過。只有抽象類的非抽象子類可以創(chuàng)建對象。
- 抽象類中不一定包含抽象方法,但是有抽象方法的類必定是抽象類。
- 抽象類中的抽象方法只是聲明,不包含方法體,就是不給出方法的具體實(shí)現(xiàn)也就是方法的具體功能。
- 構(gòu)造方法,類方法(用static修飾的方法)不能聲明為抽象方法。
- 抽象類的子類必須給出抽象類中的抽象方法的具體實(shí)現(xiàn),除非該子類也是抽象類。