詳解JAVA 抽象類
在面向?qū)ο蟮母拍钪?,所有的?duì)象都是通過(guò)類來(lái)描繪的,但是反過(guò)來(lái),并不是所有的類都是用來(lái)描繪對(duì)象的,如果一個(gè)類中沒有包含足夠的信息來(lái)描繪一個(gè)具體的對(duì)象,這樣的類就是抽象類。
抽象類除了不能實(shí)例化對(duì)象之外,類的其它功能依然存在,成員變量、成員方法和構(gòu)造方法的訪問方式和普通類一樣。
由于抽象類不能實(shí)例化對(duì)象,所以抽象類必須被繼承,才能被使用。也是因?yàn)檫@個(gè)原因,通常在設(shè)計(jì)階段決定要不要設(shè)計(jì)抽象類。
父類包含了子類集合的常見的方法,但是由于父類本身是抽象的,所以不能使用這些方法。
在Java中抽象類表示的是一種繼承關(guān)系,一個(gè)類只能繼承一個(gè)抽象類,而一個(gè)類卻可以實(shí)現(xiàn)多個(gè)接口。
抽象類
在Java語(yǔ)言中使用abstract class來(lái)定義抽象類。如下實(shí)例:
/* 文件名 : 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 類沒有什么不同,盡管該類是抽象類,但是它仍然有 3 個(gè)成員變量,7 個(gè)成員方法和 1 個(gè)構(gòu)造方法。 現(xiàn)在如果你嘗試如下的例子:
/* 文件名 : AbstractDemo.java */public class AbstractDemo{ public static void main(String [] args) { /* 以下是不允許的,會(huì)引發(fā)錯(cuò)誤 */ Employee e = new Employee('George W.', 'Houston, TX', 43); System.out.println('n Call mailCheck using Employee reference--'); e.mailCheck(); }}
當(dāng)你嘗試編譯AbstractDemo類時(shí),會(huì)產(chǎn)生如下錯(cuò)誤:
Employee.java:46: Employee is abstract; cannot be instantiated Employee e = new Employee('George W.', 'Houston, TX', 43); ^1 error
繼承抽象類
我們能通過(guò)一般的方法繼承Employee類:
/* 文件名 : 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; }}
盡管我們不能實(shí)例化一個(gè) Employee 類的對(duì)象,但是如果我們實(shí)例化一個(gè) Salary 類對(duì)象,該對(duì)象將從 Employee 類繼承 7 個(gè)成員方法,且通過(guò)該方法可以設(shè)置或獲取三個(gè)成員變量。
/* 文件名 : 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(); }}
以上程序編譯運(yùn)行結(jié)果如下:
Constructing an EmployeeConstructing an EmployeeCall mailCheck using Salary reference --Within mailCheck of Salary classMailing check to Mohd Mohtashim with salary 3600.0
Call mailCheck using Employee reference--Within mailCheck of Salary classMailing check to John Adams with salary 2400.
抽象方法
如果你想設(shè)計(jì)這樣一個(gè)類,該類包含一個(gè)特別的成員方法,該方法的具體實(shí)現(xiàn)由它的子類確定,那么你可以在父類中聲明該方法為抽象方法。
Abstract 關(guān)鍵字同樣可以用來(lái)聲明抽象方法,抽象方法只包含一個(gè)方法名,而沒有方法體。
抽象方法沒有定義,方法名后面直接跟一個(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)該抽象方法,否則,從最初的父類到最終的子類都不能用來(lái)實(shí)例化對(duì)象。
如果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ī)定
1. 抽象類不能被實(shí)例化(初學(xué)者很容易犯的錯(cuò)),如果被實(shí)例化,就會(huì)報(bào)錯(cuò),編譯無(wú)法通過(guò)。只有抽象類的非抽象子類可以創(chuàng)建對(duì)象。
2. 抽象類中不一定包含抽象方法,但是有抽象方法的類必定是抽象類。
3. 抽象類中的抽象方法只是聲明,不包含方法體,就是不給出方法的具體實(shí)現(xiàn)也就是方法的具體功能。
4. 構(gòu)造方法,類方法(用 static 修飾的方法)不能聲明為抽象方法。
5. 抽象類的子類必須給出抽象類中的抽象方法的具體實(shí)現(xiàn),除非該子類也是抽象類。
以上就是詳解JAVA 抽象類的詳細(xì)內(nèi)容,更多關(guān)于JAVA 抽象類的資料請(qǐng)關(guān)注好吧啦網(wǎng)其它相關(guān)文章!
相關(guān)文章:
1. XML入門的常見問題(一)2. CSS3中Transition屬性詳解以及示例分享3. 通過(guò)CSS數(shù)學(xué)函數(shù)實(shí)現(xiàn)動(dòng)畫特效4. 低版本IE正常運(yùn)行HTML5+CSS3網(wǎng)站的3種解決方案5. 利用CSS3新特性創(chuàng)建透明邊框三角6. 阿里前端開發(fā)中的規(guī)范要求7. CSS3實(shí)例分享之多重背景的實(shí)現(xiàn)(Multiple backgrounds)8. IE6/IE7/IE8/IE9中tbody的innerHTML不能賦值的完美解決方案9. 解析原生JS getComputedStyle10. XML入門的常見問題(二)
