詳解JAVA 抽象類
在面向?qū)ο蟮母拍钪校械膶?duì)象都是通過類來描繪的,但是反過來,并不是所有的類都是用來描繪對(duì)象的,如果一個(gè)類中沒有包含足夠的信息來描繪一個(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語言中使用abstract class來定義抽象類。如下實(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
繼承抽象類
我們能通過一般的方法繼承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è)成員方法,且通過該方法可以設(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)鍵字同樣可以用來聲明抽象方法,抽象方法只包含一個(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)該抽象方法,否則,從最初的父類到最終的子類都不能用來實(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ò),編譯無法通過。只有抽象類的非抽象子類可以創(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. 爬取今日頭條Ajax請(qǐng)求2. laravel ajax curd 搜索登錄判斷功能的實(shí)現(xiàn)3. JavaScrip簡(jiǎn)單數(shù)據(jù)類型隱式轉(zhuǎn)換的實(shí)現(xiàn)4. HTML DOM setInterval和clearInterval方法案例詳解5. 詳解JSP 內(nèi)置對(duì)象request常見用法6. ASP.NET Core按用戶等級(jí)授權(quán)的方法7. webpack高級(jí)配置與優(yōu)化詳解8. HTML <!DOCTYPE> 標(biāo)簽9. ASP動(dòng)態(tài)網(wǎng)頁制作技術(shù)經(jīng)驗(yàn)分享10. JS中map和parseInt的用法詳解
