摘要:抽象類的細(xì)節(jié)抽象類中是否有構(gòu)造函數(shù)有,用于給子類對象進(jìn)行初始化。抽象方法目的僅僅為了不讓該類創(chuàng)建對象。抽象類和接口的區(qū)別抽象類只能被繼承,而且只能單繼承。抽象類中可以定義非抽象方法,子類可以直接繼承使用。抽象類使用的是關(guān)系。
定義
好處:
提高了代碼的復(fù)用性。 讓類與類之間產(chǎn)生了關(guān)系,提供了另一個(gè)特征多態(tài)的前提。
父類的由來:
其實(shí)是由多個(gè)類不斷向上抽取共性內(nèi)容而來的。
java中對于繼承,java只支持單繼承。java雖然不直接支持多繼承,但是保留了這種多繼承機(jī)制,進(jìn)行改良。
單繼承:
一個(gè)類只能有一個(gè)父類。
多繼承:
一個(gè)類可以有多個(gè)父類。
為什么不支持多繼承呢?
因?yàn)楫?dāng)一個(gè)類同時(shí)繼承兩個(gè)父類時(shí),兩個(gè)父類中有相同的功能,那么子類對象調(diào)用該功能時(shí),運(yùn)行哪一個(gè)呢?因?yàn)楦割愔械姆椒ㄖ写嬖诜椒w。
但是java支持多重繼承。A繼承B B繼承C C繼承D。
多重繼承的出現(xiàn),就有了繼承體系。體系中的頂層父類是通過不斷向上抽取而來的。它里面定義的該體系最基本最共性內(nèi)容的功能。
所以,一個(gè)體系要想被使用,直接查閱該系統(tǒng)中的父類的功能即可知道該體系的基本用法。那么想要使用一個(gè)體系時(shí),需要建立對象。建議建立最子類對象,因?yàn)樽钭宇惒粌H可以使用父類中的功能。還可以使用子類特有的一些功能。
簡單說:
對于一個(gè)繼承體系的使用,查閱頂層父類中的內(nèi)容,創(chuàng)建最底層子類的對象。
子父類出現(xiàn)后,類中的成員都有了哪些特點(diǎn):
1:成員變量。
當(dāng)子父類中出現(xiàn)一樣的屬性時(shí),子類類型的對象,調(diào)用該屬性,值是子類的屬性值。
如果想要調(diào)用父類中的屬性值,需要使用一個(gè)關(guān)鍵字:super
This:代表是本類類型的對象引用。
Super:代表是子類所屬的父類中的內(nèi)存空間引用。
注意:子父類中通常是不會(huì)出現(xiàn)同名成員變量的,因?yàn)楦割愔兄灰x了,子類就不用在定義了,直接繼承過來用就可以了。
2:成員函數(shù)。
當(dāng)子父類中出現(xiàn)了一模一樣的方法時(shí),建立子類對象會(huì)運(yùn)行子類中的方法。好像父類中的方法被覆蓋掉一樣。所以這種情況,是函數(shù)的另一個(gè)特性:覆蓋(復(fù)寫,重寫)
什么時(shí)候使用覆蓋呢?當(dāng)一個(gè)類的功能內(nèi)容需要修改時(shí),可以通過覆蓋來實(shí)現(xiàn)。
3:構(gòu)造函數(shù)。
發(fā)現(xiàn)子類構(gòu)造函數(shù)運(yùn)行時(shí),先運(yùn)行了父類的構(gòu)造函數(shù)。為什么呢?
原因:子類的所有構(gòu)造函數(shù)中的第一行,其實(shí)都有一條隱身的語句super();
super(): 表示父類的構(gòu)造函數(shù),并會(huì)調(diào)用于參數(shù)相對應(yīng)的父類中的構(gòu)造函數(shù)。而super():是在調(diào)用父類中空參數(shù)的構(gòu)造函數(shù)。
為什么子類對象初始化時(shí),都需要調(diào)用父類中的函數(shù)?(為什么要在子類構(gòu)造函數(shù)的第一行加入這個(gè)super()?)
因?yàn)樽宇惱^承父類,會(huì)繼承到父類中的數(shù)據(jù),所以必須要看父類是如何對自己的數(shù)據(jù)進(jìn)行初始化的。所以子類在進(jìn)行對象初始化時(shí),先調(diào)用父類的構(gòu)造函數(shù),這就是子類的實(shí)例化過程。
注意:子類中所有的構(gòu)造函數(shù)都會(huì)默認(rèn)訪問父類中的空參數(shù)的構(gòu)造函數(shù),因?yàn)槊恳粋€(gè)子類構(gòu)造內(nèi)第一行都有默認(rèn)的語句super();
如果父類中沒有空參數(shù)的構(gòu)造函數(shù),那么子類的構(gòu)造函數(shù)內(nèi),必須通過super語句指定要訪問的父類中的構(gòu)造函數(shù)。
如果子類構(gòu)造函數(shù)中用this來指定調(diào)用子類自己的構(gòu)造函數(shù),那么被調(diào)用的構(gòu)造函數(shù)也一樣會(huì)訪問父類中的構(gòu)造函數(shù)。
問題:super()和this()是否可以同時(shí)出現(xiàn)的構(gòu)造函數(shù)中。
兩個(gè)語句只能有一個(gè)定義在第一行,所以只能出現(xiàn)其中一個(gè)。
super()或者this():為什么一定要定義在第一行?
因?yàn)閟uper()或者this()都是調(diào)用構(gòu)造函數(shù),構(gòu)造函數(shù)用于初始化,所以初始化的動(dòng)作要先完成。
繼承的細(xì)節(jié):
什么時(shí)候使用繼承呢?
當(dāng)類與類之間存在著所屬關(guān)系時(shí),才具備了繼承的前提。a是b中的一種。a繼承b。狼是犬科中的一種。
英文書中,所屬關(guān)系:" is a "
注意:不要僅僅為了獲取其他類中的已有成員進(jìn)行繼承。
所以判斷所屬關(guān)系,可以簡單看,如果繼承后,被繼承的類中的功能,都可以被該子類所具備,那么繼承成立。如果不是,不可以繼承。
細(xì)節(jié)二:
在方法覆蓋時(shí),注意兩點(diǎn):
1:子類覆蓋父類時(shí),必須要保證,子類方法的權(quán)限必須大于等于父類方法權(quán)限可以實(shí)現(xiàn)繼承。否則,編譯失敗。
2:覆蓋時(shí),要么都靜態(tài),要么都不靜態(tài)。 (靜態(tài)只能覆蓋靜態(tài),或者被靜態(tài)覆蓋)
繼承的一個(gè)弊端:打破了封裝性。對于一些類,或者類中功能,是需要被繼承,或者復(fù)寫的。
這時(shí)如何解決問題呢?介紹一個(gè)關(guān)鍵字,final:最終。
final特點(diǎn):
1:這個(gè)關(guān)鍵字是一個(gè)修飾符,可以修飾類,方法,變量。
2:被final修飾的類是一個(gè)最終類,不可以被繼承。
3:被final修飾的方法是一個(gè)最終方法,不可以被覆蓋。
4:被final修飾的變量是一個(gè)常量,只能賦值一次。
其實(shí)這樣的原因的就是給一些固定的數(shù)據(jù)起個(gè)閱讀性較強(qiáng)的名稱。
不加final修飾不是也可以使用嗎?那么這個(gè)值是一個(gè)變量,是可以更改的。加了final,程序更為嚴(yán)謹(jǐn)。常量名稱定義時(shí),有規(guī)范,所有字母都大寫,如果由多個(gè)單詞組成,中間用 _ 連接。
抽象類: abstract
抽象:不具體,看不明白。抽象類表象體現(xiàn)。
在不斷抽取過程中,將共性內(nèi)容中的方法聲明抽取,但是方法不一樣,沒有抽取,這時(shí)抽取到的方法,并不具體,需要被指定關(guān)鍵字abstract所標(biāo)示,聲明為抽象方法。
抽象方法所在類一定要標(biāo)示為抽象類,也就是說該類需要被abstract關(guān)鍵字所修飾。
抽象類的特點(diǎn):
1:抽象方法只能定義在抽象類中,抽象類和抽象方法必須由abstract關(guān)鍵字修飾(可以描述類和方法,不可以描述變量)。
2:抽象方法只定義方法聲明,并不定義方法實(shí)現(xiàn)。
3:抽象類不可以被創(chuàng)建對象(實(shí)例化)。
4:只有通過子類繼承抽象類并覆蓋了抽象類中的所有抽象方法后,該子類才可以實(shí)例化。否則,該子類還是一個(gè)抽象類。
抽象類的細(xì)節(jié):
1:抽象類中是否有構(gòu)造函數(shù)?有,用于給子類對象進(jìn)行初始化。
2:抽象類中是否可以定義非抽象方法?
可以。其實(shí),抽象類和一般類沒有太大的區(qū)別,都是在描述事物,只不過抽象類在描述事物時(shí),有些功能不具體。所以抽象類和一般類在定義上,都是需要定義屬性和行為的。只不過,比一般類多了一個(gè)抽象函數(shù)。而且比一般類少了一個(gè)創(chuàng)建對象的部分。
3:抽象關(guān)鍵字abstract和哪些不可以共存?final ,private , static
4:抽象類中可不可以不定義抽象方法?可以。抽象方法目的僅僅為了不讓該類創(chuàng)建對象。
模板方法設(shè)計(jì)模式:
解決的問題:當(dāng)功能內(nèi)部一部分實(shí)現(xiàn)時(shí)確定,一部分實(shí)現(xiàn)是不確定的。這時(shí)可以把不確定的部分暴露出去,讓子類去實(shí)現(xiàn)。
abstract class GetTime{
public final void getTime(){ //此功能如果不需要復(fù)寫,可加final限定
long start = System.currentTimeMillis();
code(); //不確定的功能部分,提取出來,通過抽象方法實(shí)現(xiàn)
long end = System.currentTimeMillis();
System.out.println("毫秒是:"+(end-start));
}
public abstract void code(); //抽象不確定的功能,讓子類復(fù)寫實(shí)現(xiàn)
}
class SubDemo extends GetTime{
public void code(){ //子類復(fù)寫功能方法
for(int y=0; y<1000; y++){
System.out.println("y");
}
}
}
接 口:★★★★★
1:是用關(guān)鍵字interface定義的。
2:接口中包含的成員,最常見的有全局常量、抽象方法。
注意:接口中的成員都有固定的修飾符。
成員變量:public static final
成員方法:public abstract
interface Inter{
public static final int x = 3;
public abstract void show();
}
3:接口中有抽象方法,說明接口不可以實(shí)例化。接口的子類必須實(shí)現(xiàn)了接口中所有的抽象方法后,該子類才可以實(shí)例化。否則,該子類還是一個(gè)抽象類。
4:類與類之間存在著繼承關(guān)系,類與接口中間存在的是實(shí)現(xiàn)關(guān)系。
繼承用extends ;實(shí)現(xiàn)用implements ;
5:接口和類不一樣的地方,就是,接口可以被多實(shí)現(xiàn),這就是多繼承改良后的結(jié)果。java將多繼承機(jī)制通過多現(xiàn)實(shí)來體現(xiàn)。
6:一個(gè)類在繼承另一個(gè)類的同時(shí),還可以實(shí)現(xiàn)多個(gè)接口。所以接口的出現(xiàn)避免了單繼承的局限性。還可以將類進(jìn)行功能的擴(kuò)展。
7:其實(shí)java中是有多繼承的。接口與接口之間存在著繼承關(guān)系,接口可以多繼承接口。
接口都用于設(shè)計(jì)上,設(shè)計(jì)上的特點(diǎn):(可以理解主板上提供的接口)
1:接口是對外提供的規(guī)則。
2:接口是功能的擴(kuò)展。
3:接口的出現(xiàn)降低了耦合性。
抽象類與接口:
抽象類:一般用于描述一個(gè)體系單元,將一組共性內(nèi)容進(jìn)行抽取,特點(diǎn):可以在類中定義抽象內(nèi)容讓子類實(shí)現(xiàn),可以定義非抽象內(nèi)容讓子類直接使用。它里面定義的都是一些體系中的基本內(nèi)容。
接口:一般用于定義對象的擴(kuò)展功能,是在繼承之外還需這個(gè)對象具備的一些功能。
抽象類和接口的共性:都是不斷向上抽取的結(jié)果。
抽象類和接口的區(qū)別:
1:抽象類只能被繼承,而且只能單繼承。
接口需要被實(shí)現(xiàn),而且可以多實(shí)現(xiàn)。
2:抽象類中可以定義非抽象方法,子類可以直接繼承使用。
接口中都有抽象方法,需要子類去實(shí)現(xiàn)。
3:抽象類使用的是 is a 關(guān)系。
接口使用的 like a 關(guān)系。
4:抽象類的成員修飾符可以自定義。
接口中的成員修飾符是固定的。全都是public的。
在開發(fā)之前,先定義規(guī)則,A和B分別開發(fā),A負(fù)責(zé)實(shí)現(xiàn)這個(gè)規(guī)則,B負(fù)責(zé)使用這個(gè)規(guī)則。至于A是如何對規(guī)則具體實(shí)現(xiàn)的,B是不需要知道的。這樣這個(gè)接口的出現(xiàn)就降低了A和B直接耦合性。
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://www.ezyhdfw.cn/yun/73190.html
摘要:靜態(tài)塊代碼初始化其實(shí),整個(gè)靜態(tài)代碼塊可以看作是一個(gè)靜態(tài)成員。和普通的非靜態(tài)成員初始化一樣,它的執(zhí)行也發(fā)生在構(gòu)造器調(diào)用之前,并且每當(dāng)創(chuàng)建對象之前都會(huì)調(diào)用??诶^承中涉及的初始化大的原則是沒有父類,就沒子類。 這個(gè)教程,咱們來對Java中設(shè)計(jì)到的初始化規(guī)則,或者說初始化順序,來做一下匯總,這里我基本上把Java中,默認(rèn)初始化,靜態(tài)成員初始化,非靜態(tài)成員初始化,靜態(tài)代碼塊,非靜態(tài)代碼塊,以及繼...
摘要:摘要今年的先知白帽大會(huì),與會(huì)者將能夠親身感受到非常多有趣的技術(shù)議題,如在國際賽事中屢奪佳績的團(tuán)隊(duì),其隊(duì)長將親臨現(xiàn)場,分享穿針引線般的漏洞利用藝術(shù)。從數(shù)據(jù)視角探索安全威脅阿里云安全工程師議題解讀本議題討論了數(shù)據(jù)為安全人員思維方式帶來的變化。 摘要: 今年的先知白帽大會(huì),與會(huì)者將能夠親身感受到非常多有趣的技術(shù)議題,如HITCON在國際賽事中屢奪佳績的CTF團(tuán)隊(duì),其隊(duì)長Orange將親臨現(xiàn)場...
摘要:面向?qū)ο笈c面向過程的區(qū)別要知道,二者并不是非此即彼,而是相輔相成的。而面向過程,則在微觀上對對象內(nèi)部進(jìn)行具體的實(shí)現(xiàn)。面向?qū)ο蟮娜筇匦哉f到面向?qū)ο?,就不得不說其三大特性封裝繼承和多態(tài)。封裝封裝是面向?qū)ο笞罨A(chǔ)的特性。 作者:伯特出處:github.com/ruicbAndroid/LoulanPlan聲明:本文出自伯特的《LoulanPlan》,轉(zhuǎn)載務(wù)必注明作者及出處。 剛學(xué)習(xí) Jav...
摘要:眾多面向?qū)ο蟮木幊趟枷腚m不盡一致,但是無論哪種面向?qū)ο缶幊陶Z言都具有以下的共通功能。原型編程以類為中心的傳統(tǒng)面向?qū)ο缶幊?,是以類為基礎(chǔ)生成新對象。而原型模式的面向?qū)ο缶幊陶Z言沒有類這樣一個(gè)概念。 什么是面向?qū)ο螅窟@個(gè)問題往往會(huì)問到剛畢業(yè)的新手or實(shí)習(xí)生上,也是往往作為一個(gè)技術(shù)面試的開頭題。在這里我們不去談如何答(fu)好(yan)問(guo)題(qu),僅談?wù)勎宜斫獾拿嫦驅(qū)ο蟆?從歷...
摘要:知識(shí)點(diǎn)總結(jié)面向?qū)ο笾R(shí)點(diǎn)總結(jié)面向?qū)ο竺嫦驅(qū)ο蟾拍钍窍鄬τ诿嫦蜻^程而言,過程其實(shí)就是函數(shù),對象是將函數(shù)和屬性進(jìn)行了封裝。指向了該對象關(guān)鍵字代表對象。靜態(tài)變量所屬于類,所以也稱為類變量成員變量存在于堆內(nèi)存中。 Java知識(shí)點(diǎn)總結(jié)(面向?qū)ο螅?@(Java知識(shí)點(diǎn)總結(jié))[Java, Java面向?qū)ο骫 [toc] 面向?qū)ο蟾拍?是相對于面向過程而言,過程其實(shí)就是函數(shù),對象是將函數(shù)和屬性進(jìn)行了封...
閱讀 3658·2023-04-26 00:05
閱讀 1020·2021-11-11 16:55
閱讀 3661·2021-09-26 09:46
閱讀 3596·2019-08-30 15:56
閱讀 972·2019-08-30 15:55
閱讀 2988·2019-08-30 15:53
閱讀 2020·2019-08-29 17:11
閱讀 868·2019-08-29 16:52