Java抽象類(lèi)和抽象方法定義與用法實(shí)例詳解
本文實(shí)例講述了Java抽象類(lèi)和抽象方法定義與用法。分享給大家供大家參考,具體如下:
一、Java抽象類(lèi)
參考資料:Java抽象類(lèi) 詳解
1、抽象類(lèi)的說(shuō)明
在面向?qū)ο蟮母拍钪?,所有的?duì)象都是通過(guò)類(lèi)來(lái)描繪的,但是反過(guò)來(lái),并不是所有的類(lèi)都是用來(lái)描繪對(duì)象的,如果一個(gè)類(lèi)中沒(méi)有包含足夠的信息來(lái)描繪一個(gè)具體的對(duì)象,這樣的類(lèi)就是抽象類(lèi)。
抽象類(lèi)除了不能實(shí)例化對(duì)象之外,類(lèi)的其它功能依然存在,成員變量、成員方法和構(gòu)造方法的訪問(wèn)方式和普通類(lèi)一樣。
由于抽象類(lèi)不能實(shí)例化對(duì)象,所以抽象類(lèi)必須被繼承,才能被使用。也是因?yàn)檫@個(gè)原因,通常在設(shè)計(jì)階段決定要不要設(shè)計(jì)抽象類(lèi)。
父類(lèi)包含了子類(lèi)集合的常見(jiàn)的方法,但是由于父類(lèi)本身是抽象的,所以不能使用這些方法。
在Java中抽象類(lèi)表示的是一種繼承關(guān)系,一個(gè)類(lèi)只能繼承一個(gè)抽象類(lèi),而一個(gè)類(lèi)卻可以實(shí)現(xiàn)多個(gè)接口。
在Java語(yǔ)言中使用abstract class來(lái)定義抽象類(lèi)。
抽象類(lèi)是為了把相同的但不確定的東西的提取出來(lái),為了以后的重用。定義成抽象類(lèi)的目的,就是為了在子類(lèi)中實(shí)現(xiàn)抽象類(lèi)。
2、抽象類(lèi)的定義
abstract class A{//定義一個(gè)抽象類(lèi)
public void fun(){//普通方法
System.out.println("存在方法體的方法");
}
//不存在方法體的抽象方法
public abstract void print();//抽象方法,沒(méi)有方法體,有abstract關(guān)鍵字做修飾
}
3、抽象類(lèi)的使用
抽象類(lèi)的使用原則如下:
(1)抽象方法必須為public或者protected(因?yàn)槿绻麨閜rivate,則不能被子類(lèi)繼承,子類(lèi)便無(wú)法實(shí)現(xiàn)該方法),缺省情況下默認(rèn)為public;
(2)抽象類(lèi)不能直接實(shí)例化,需要依靠子類(lèi)采用向上轉(zhuǎn)型的方式處理;
(3)抽象類(lèi)必須有子類(lèi),使用extends繼承,一個(gè)子類(lèi)只能繼承一個(gè)抽象類(lèi);
(4)子類(lèi)(如果不是抽象類(lèi))則必須覆寫(xiě)抽象類(lèi)之中的全部抽象方法(如果子類(lèi)沒(méi)有實(shí)現(xiàn)父類(lèi)的抽象方法,則必須將子類(lèi)也定義為為abstract類(lèi)。);
package com.wz.abstractdemo;
abstract class A{//定義一個(gè)抽象類(lèi)
public void fun(){//普通方法
System.out.println("存在方法體的方法");
}
public abstract void print();//抽象方法,沒(méi)有方法體,有abstract關(guān)鍵字做修飾
}
//單繼承
class B extends A{//B類(lèi)是抽象類(lèi)的子類(lèi),是一個(gè)普通類(lèi)
@Override
public void print() {//強(qiáng)制要求覆寫(xiě)
System.out.println("Hello World !");
}
}
public class TestDemo {
public static void main(String[] args) {
A a = new B();//向上轉(zhuǎn)型
a.fun();//被子類(lèi)所覆寫(xiě)的過(guò)的方法
}
}
(1)抽象類(lèi)的子類(lèi)里面有明確的方法覆寫(xiě)要求,而普通類(lèi)可以有選擇性的來(lái)決定是否需要覆寫(xiě);
(2)抽象類(lèi)實(shí)際上就比普通類(lèi)多了一些抽象方法而已,其他組成部分和普通類(lèi)完全一樣;
(3)普通類(lèi)對(duì)象可以直接實(shí)例化,但抽象類(lèi)的對(duì)象必須經(jīng)過(guò)向上轉(zhuǎn)型之后才可以得到。
雖然一個(gè)類(lèi)的子類(lèi)可以去繼承任意的一個(gè)普通類(lèi),可是從開(kāi)發(fā)的實(shí)際要求來(lái)講,普通類(lèi)盡量不要去繼承另外一個(gè)普通類(lèi),而是去繼承抽象類(lèi)。
4、抽象類(lèi)的使用限制
(1)抽象類(lèi)中有構(gòu)造方法么?
由于抽象類(lèi)里會(huì)存在一些屬性,那么抽象類(lèi)中一定存在構(gòu)造方法,其存在目的是為了屬性的初始化。
并且子類(lèi)對(duì)象實(shí)例化的時(shí)候,依然滿足先執(zhí)行父類(lèi)構(gòu)造,再執(zhí)行子類(lèi)構(gòu)造的順序。
范例如下:
package com.wz.abstractdemo;
abstract class A{//定義一個(gè)抽象類(lèi)
public A(){
System.out.println("*****A類(lèi)構(gòu)造方法*****");
}
public abstract void print();//抽象方法,沒(méi)有方法體,有abstract關(guān)鍵字做修飾
}
//單繼承
class B extends A{//B類(lèi)是抽象類(lèi)的子類(lèi),是一個(gè)普通類(lèi)
public B(){
System.out.println("*****B類(lèi)構(gòu)造方法*****");
}
@Override
public void print() {//強(qiáng)制要求覆寫(xiě)
System.out.println("Hello World !");
}
}
public class TestDemo {
public static void main(String[] args) {
A a = new B();//向上轉(zhuǎn)型
}
}
執(zhí)行結(jié)果:
*****A類(lèi)構(gòu)造方法*****
*****B類(lèi)構(gòu)造方法*****
(2)抽象類(lèi)可以用final聲明么?
不能,因?yàn)槌橄箢?lèi)必須有子類(lèi),而final定義的類(lèi)不能有子類(lèi);
二、Java抽象方法
參考資料:JAVA抽象類(lèi)和抽象方法
用abstract修飾的方法,即抽象方法。并且,抽象方法不能有方法主體。格式如下:
abstract void xxx();
說(shuō)明:
- 一旦類(lèi)中包含了abstract抽象方法,那類(lèi)該類(lèi)必須聲明為abstract類(lèi)。
- 抽象類(lèi)中不一定要包含abstrace方法。
- 抽象類(lèi)不能被實(shí)例化。因?yàn)槌橄箢?lèi)中方法未具體化,這是一種不完整的類(lèi),所以直接實(shí)例化也就沒(méi)有意義了。
- 任何子類(lèi)必須重寫(xiě)父類(lèi)的抽象方法,或者聲明自身為抽象類(lèi)
范例:bstract關(guān)鍵字可以用來(lái)聲明抽象方法,抽象方法只包含一個(gè)方法名,而沒(méi)有方法體。
抽象方法沒(méi)有定義,方法名后面直接跟一個(gè)分號(hào),而不是花括號(hào)。
public abstract class Employee
{
private String name;
private String address;
private int number;
public abstract double computePay();
//其余代碼
}
如果Salary類(lèi)繼承了上面的Employee抽象類(lèi),那么它必須實(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;
}
//其余代碼
}
【抽象類(lèi)總結(jié)規(guī)定】
- 抽象類(lèi)不能被實(shí)例化(初學(xué)者很容易犯的錯(cuò)),如果被實(shí)例化,就會(huì)報(bào)錯(cuò),編譯無(wú)法通過(guò)。只有抽象類(lèi)的非抽象子類(lèi)可以創(chuàng)建對(duì)象。
- 抽象類(lèi)中不一定包含抽象方法,但是有抽象方法的類(lèi)必定是抽象類(lèi)。
- 抽象類(lèi)中的抽象方法只是聲明,不包含方法體,就是不給出方法的具體實(shí)現(xiàn)也就是方法的具體功能。
- 構(gòu)造方法,類(lèi)方法(用static修飾的方法)不能聲明為抽象方法。
- 抽象類(lèi)的子類(lèi)必須給出抽象類(lèi)中的抽象方法的具體實(shí)現(xiàn),除非該子類(lèi)也是抽象類(lèi)。
更多java相關(guān)內(nèi)容感興趣的讀者可查看本站專(zhuān)題:《Java面向?qū)ο蟪绦蛟O(shè)計(jì)入門(mén)與進(jìn)階教程》、《Java數(shù)據(jù)結(jié)構(gòu)與算法教程》、《Java操作DOM節(jié)點(diǎn)技巧總結(jié)》、《Java文件與目錄操作技巧匯總》和《Java緩存操作技巧匯總》
希望本文所述對(duì)大家java程序設(shè)計(jì)有所幫助。
相關(guān)文章
Java入門(mén)絆腳石之Override和Overload的區(qū)別詳解
重寫(xiě)是子類(lèi)對(duì)父類(lèi)的允許訪問(wèn)的方法的實(shí)現(xiàn)過(guò)程進(jìn)行重新編寫(xiě), 返回值和形參都不能改變。即外殼不變,核心重寫(xiě)!重寫(xiě)的好處在于子類(lèi)可以根據(jù)需要,定義特定于自己的行為。重載是在一個(gè)類(lèi)里面,方法名字相同,而參數(shù)不同。返回類(lèi)型可以相同也可以不同2021-10-10
java中1+1d/5和1+1/5的區(qū)別說(shuō)明
這篇文章主要介紹了java中1+1d/5和1+1/5的區(qū)別說(shuō)明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-10-10
Maven默認(rèn)使用JDK1.5的問(wèn)題及解決方案
這篇文章主要介紹了Maven默認(rèn)使用JDK1.5的問(wèn)題及解決方案,本文給大家分享兩種方式,通過(guò)圖文并茂的形式給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-04-04
java通過(guò)注解翻譯字典的實(shí)現(xiàn)示例
本文主要介紹了java通過(guò)注解翻譯字典的實(shí)現(xiàn)示例,文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-04-04
Java 8中Collectors.toMap空指針異常源碼解析
這篇文章主要為大家介紹了Java 8中Collectors.toMap空指針異常源碼解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-08-08
Java的RocketMq水平擴(kuò)展及負(fù)載均衡詳解
這篇文章主要介紹了Java的RocketMq水平擴(kuò)展及負(fù)載均衡詳解,RocketMQ是一個(gè)分布式具有高度可擴(kuò)展性的消息中間件,本文旨在探索在broker端,生產(chǎn)端,以及消費(fèi)端是如何做到橫向擴(kuò)展以及負(fù)載均衡的,需要的朋友可以參考下2024-01-01
Java AbstractMethodError案例分析詳解
這篇文章主要介紹了Java AbstractMethodError案例分析詳解,本篇文章通過(guò)簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-08-08

