java基礎(chǔ)知識總結(jié)_第1頁
java基礎(chǔ)知識總結(jié)_第2頁
java基礎(chǔ)知識總結(jié)_第3頁
java基礎(chǔ)知識總結(jié)_第4頁
java基礎(chǔ)知識總結(jié)_第5頁
已閱讀5頁,還剩21頁未讀, 繼續(xù)免費(fèi)閱讀

下載本文檔

版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請進(jìn)行舉報或認(rèn)領(lǐng)

文檔簡介

#/64privateintage;//私有的訪問權(quán)限最低,只有在本類中的訪問有效。注意:私有僅僅是封裝的一種體現(xiàn)形式而已。私有的成員:其他類不能直接創(chuàng)建對象訪問,所以只有通過本類對外提供具體的訪問方式來完成對私有的訪問,可以通過對外提供函數(shù)的形式對其進(jìn)行訪問。好處:可以在函數(shù)中加入邏輯判斷等操作,對數(shù)據(jù)進(jìn)行判斷等操作??偨Y(jié):開發(fā)時,記住,屬性是用于存儲數(shù)據(jù)的,直接被訪問,容易出現(xiàn)安全隱患,所以,類中的屬性通常被私有化,并對外提供公共的訪問方法。這個方法一般有兩個,規(guī)范寫法:對于屬性xxx,可以使用setXXX(),getXXX()對其進(jìn)行操作。類中怎么沒有定義主函數(shù)呢?注意:主函數(shù)的存在,僅為該類是否需要獨(dú)立運(yùn)行,如果不需要,主函數(shù)是不用定義的。主函數(shù)的解釋:保證所在類的獨(dú)立運(yùn)行,是程序的入口,被jvm調(diào)用。成員變量和局部變量的區(qū)別:1:成員變量直接定義在類中。局部變量定義在方法中,參數(shù)上,語句中。2:成員變量在這個類中有效。局部變量只在自己所屬的大括號內(nèi)有效,大括號結(jié)束,局部變量失去作用域。3:成員變量存在于堆內(nèi)存中,隨著對象的產(chǎn)生而存在,消失而消失。局部變量存在于棧內(nèi)存中,隨著所屬區(qū)域的運(yùn)行而存在,結(jié)束而釋放。構(gòu)造函數(shù):用于給對象進(jìn)行初始化,是給與之對應(yīng)的對象進(jìn)行初始化,它具有針對性,函數(shù)中的一種。特點(diǎn):1:該函數(shù)的名稱和所在類的名稱相同。2:不需要定義返回值類型。3:該函數(shù)沒有具體的返回值。記?。核袑ο髣?chuàng)建時,都需要初始化才可以使用。注意事項:一個類在定義時,如果沒有定義過構(gòu)造函數(shù),那么該類中會自動生成一個空參數(shù)的構(gòu)造函數(shù),為了方便該類創(chuàng)建對象,完成初始化。如果在類中自定義了構(gòu)造函數(shù),那么默認(rèn)的構(gòu)造函數(shù)就沒有了。一個類中,可以有多個構(gòu)造函數(shù),因為它們的函數(shù)名稱都相同,所以只能通過參數(shù)列表來區(qū)分。所以,一個類中如果出現(xiàn)多個構(gòu)造函數(shù)。它們的存在是以重載體現(xiàn)的。構(gòu)造函數(shù)和一般函數(shù)有什么區(qū)別呢?1:兩個函數(shù)定義格式不同。2:構(gòu)造函數(shù)是在對象創(chuàng)建時,就被調(diào)用,用于初始化,而且初始化動作只執(zhí)行一次。一般函數(shù),是對象創(chuàng)建后,需要調(diào)用才執(zhí)行,可以被調(diào)用多次。什么時候使用構(gòu)造函數(shù)呢?分析事物時,發(fā)現(xiàn)具體事物一出現(xiàn),就具備了一些特征,那就將這些特征定義到構(gòu)造函數(shù)內(nèi)。構(gòu)造代碼塊和構(gòu)造函數(shù)有什么區(qū)別?構(gòu)造代碼塊:是給所有的對象進(jìn)行初始化,也就是說,所有的對象都會調(diào)用一個代碼塊。只要對象一建立。就會調(diào)用這個代碼塊。構(gòu)造函數(shù):是給與之對應(yīng)的對象進(jìn)行初始化。它具有針對性。Personp=newPerson();創(chuàng)建一個對象都在內(nèi)存中做了什么事情?1:先將硬盤上指定位置的Person.class文件加載進(jìn)內(nèi)存。2:執(zhí)行main方法時,在棧內(nèi)存中開辟了main方法的空間(壓棧-進(jìn)棧),然后在main方法的棧區(qū)分配了一個變量P。3:在堆內(nèi)存中開辟一個實(shí)體空間,分配了一個內(nèi)存首地址值。new4:在該實(shí)體空間中進(jìn)行屬性的空間分配,并進(jìn)行了默認(rèn)初始化。5:對空間中的屬性進(jìn)行顯示初始化。6:進(jìn)行實(shí)體的構(gòu)造代碼塊初始化。7:調(diào)用該實(shí)體對應(yīng)的構(gòu)造函數(shù),進(jìn)行構(gòu)造函數(shù)初始化。()8:將首地址賦值給p,p變量就引用了該實(shí)體。(指向了該對象)封裝(面向?qū)ο筇卣髦唬菏侵鸽[藏對象的屬性和實(shí)現(xiàn)細(xì)節(jié),僅對外提供公共訪問方式。好處:將變化隔離;便于使用;提高重用性;安全性。封裝原則:將不需要對外提供的內(nèi)容都隱藏起來,把屬性都隱藏,提供公共方法對其訪問。this:代表對象。就是所在函數(shù)所屬對象的引用。this到底代表什么呢?哪個對象調(diào)用了this所在的函數(shù),this就代表哪個對象,就是哪個對象的引用。開發(fā)時,什么時候使用this呢?在定義功能時,如果該功能內(nèi)部使用到了調(diào)用該功能的對象,這時就用this來表示這個對象。this還可以用于構(gòu)造函數(shù)間的調(diào)用。調(diào)用格式:this(實(shí)際參數(shù));this對象后面跟上.調(diào)用的是成員屬性和成員方法(一般方法);this對象后面跟上()調(diào)用的是本類中的對應(yīng)參數(shù)的構(gòu)造函數(shù)。注意:用this調(diào)用構(gòu)造函數(shù),必須定義在構(gòu)造函數(shù)的第一行。因為構(gòu)造函數(shù)是用于初始化的,所以初始化動作一定要執(zhí)行。否則編譯失敗。static:***關(guān)鍵字,是一個修飾符,用于修飾成員(成員變量和成員函數(shù))。特點(diǎn):1,想要實(shí)現(xiàn)對象中的共性數(shù)據(jù)的對象共享??梢詫⑦@個數(shù)據(jù)進(jìn)行靜態(tài)修飾。2,被靜態(tài)修飾的成員,可以直接被類名所調(diào)用。也就是說,靜態(tài)的成員多了一種調(diào)用方式。類名.靜態(tài)方式。3,靜態(tài)隨著類的加載而加載。而且優(yōu)先于對象存在。弊端:1,有些數(shù)據(jù)是對象特有的數(shù)據(jù),是不可以被靜態(tài)修飾的。因為那樣的話,特有數(shù)據(jù)會變成對象的共享數(shù)據(jù)。這樣對事物的描述就出了問題。所以,在定義靜態(tài)時,必須要明確,這個數(shù)據(jù)是否是被對象所共享的。2,靜態(tài)方法只能訪問靜態(tài)成員,不可以訪問非靜態(tài)成員。因為靜態(tài)方法加載時,優(yōu)先于對象存在,所以沒有辦法訪問對象中的成員。3,靜態(tài)方法中不能使用this,super關(guān)鍵字。因為this代表對象,而靜態(tài)在時,有可能沒有對象,所以this無法使用。4,主函數(shù)是靜態(tài)的。什么時候定義靜態(tài)成員呢?或者說:定義成員時,到底需不需要被靜態(tài)修飾呢?成員分兩種:1,成員變量。(數(shù)據(jù)共享時靜態(tài)化)該成員變量的數(shù)據(jù)是否是所有對象都一樣:如果是,那么該變量需要被靜態(tài)修飾,因為是共享的數(shù)據(jù)。如果不是,那么就說這是對象的特有數(shù)據(jù),要存儲到對象中。2,成員函數(shù)。(方法中沒有調(diào)用特有數(shù)據(jù)時就定義成靜態(tài))如果判斷成員函數(shù)是否需要被靜態(tài)修飾呢?只要參考,該函數(shù)內(nèi)是否訪問了對象中的特有數(shù)據(jù):如果有訪問特有數(shù)據(jù),那方法不能被靜態(tài)修飾。如果沒有訪問過特有數(shù)據(jù),那么這個方法需要被靜態(tài)修飾。成員變量和靜態(tài)變量的區(qū)別:1,成員變量所屬于對象。所以也稱為實(shí)例變量。靜態(tài)變量所屬于類。所以也稱為類變量。2,成員變量存在于堆內(nèi)存中。靜態(tài)變量存在于方法區(qū)中。3,成員變量隨著對象創(chuàng)建而存在。隨著對象被回收而消失。靜態(tài)變量隨著類的加載而存在。隨著類的消失而消失。4,成員變量只能被對象所調(diào)用。靜態(tài)變量可以被對象調(diào)用,也可以被類名調(diào)用。所以,成員變量可以稱為對象的特有數(shù)據(jù),靜態(tài)變量稱為對象的共享數(shù)據(jù)。靜態(tài)的注意:靜態(tài)的生命周期很長。靜態(tài)代碼塊:就是一個有靜態(tài)關(guān)鍵字標(biāo)示的一個代碼塊區(qū)域。定義在類中。作用:可以完成類的初始化。靜態(tài)代碼塊隨著類的加載而執(zhí)行,而且只執(zhí)行一次(new多個對象就只執(zhí)行一次)。如果和主函數(shù)在同一類中,優(yōu)先于主函數(shù)執(zhí)行。Public:訪問權(quán)限最大。static:不需要對象,直接類名即可。void:主函數(shù)沒有返回值。Main:主函數(shù)特定的名稱。(String[]args):主函數(shù)的參數(shù),是一個字符串?dāng)?shù)組類型的參數(shù),jvm調(diào)用main方法時,傳遞的實(shí)際參數(shù)是newString[0]。jvm默認(rèn)傳遞的是長度為0的字符串?dāng)?shù)組,我們在運(yùn)行該類時,也可以指定具體的參數(shù)進(jìn)行傳遞??梢栽诳刂婆_,運(yùn)行該類時,在后面加入?yún)?shù)。參數(shù)之間通過空格隔開。jvm會自動將這些字符串參數(shù)作為args數(shù)組中的元素,進(jìn)行存儲。靜態(tài)代碼塊、構(gòu)造代碼塊、構(gòu)造函數(shù)同時存在時的執(zhí)行順序:靜態(tài)代碼塊9構(gòu)造代碼塊9構(gòu)造函數(shù);生成Java幫助文檔:命令格式:javadoc-d文件夾名-auther-version*.java/**//格式*類描述*@author作者名*@version版本號*//***方法描述*@param參數(shù)描述*@return返回值描述*/設(shè)計模式:解決問題最行之有效的思想。是一套被反復(fù)使用、多數(shù)人知曉的、經(jīng)過分類編目的、代碼設(shè)計經(jīng)驗的總結(jié)。使用設(shè)計模式是為了可重用代碼、讓代碼更容易被他人理解、保證代碼可靠性。java中有23種設(shè)計模式:單例設(shè)計模式:★★★★★解決的問題:保證一個類在內(nèi)存中的對象唯一性。比如:多程序讀取一個配置文件時,建議配置文件封裝成對象。會方便操作其中數(shù)據(jù),又要保證多個程序讀到的是同一個配置文件對象,就需要該配置文件對象在內(nèi)存中是唯一的。RuntimeO方法就是單例設(shè)計模式進(jìn)行設(shè)計的。如何保證對象唯一性呢?思想:1,不讓其他程序創(chuàng)建該類對象。2,在本類中創(chuàng)建一個本類對象。3,對外提供方法,讓其他程序獲取這個對象。步驟:1,因為創(chuàng)建對象都需要構(gòu)造函數(shù)初始化,只要將本類中的構(gòu)造函數(shù)私有化,其他程序就無法再創(chuàng)建該類對象;2,就在類中創(chuàng)建一個本類的對象;3,定義一個方法,返回該對象,讓其他程序可以通過方法就得到本類對象。(作用:可控)代碼體現(xiàn):1,私有化構(gòu)造函數(shù);2,創(chuàng)建私有并靜態(tài)的本類對象;3,定義公有并靜態(tài)的方法,返回該對象。//餓漢式classSingle{privateSingle(){}//私有化構(gòu)造函數(shù)。privatestaticSingles=newSingle();//創(chuàng)建私有并靜態(tài)的本類對象。publicstaticSinglegetInstance(){//定義公有并靜態(tài)的方法,返回該對象returns;}}//懶漢式:延遲加載方式。classSingle2{privateSingle2(){}privatestaticSingle2s=null;publicstaticSingle2getInstance(){if(s==null)s=newSingle2();returns;}}繼承(面向?qū)ο筇卣髦唬┖锰帲?:提高了代碼的復(fù)用性。2:讓類與類之間產(chǎn)生了關(guān)系,提供了另一個特征多態(tài)的前提。父類的由來:其實(shí)是由多個類不斷向上抽取共性內(nèi)容而來的。java中對于繼承,java只支持單繼承。java雖然不直接支持多繼承,但是保留了這種多繼承機(jī)制,進(jìn)行改良。單繼承:一個類只能有一個父類。多繼承:一個類可以有多個父類。為什么不支持多繼承呢?因為當(dāng)一個類同時繼承兩個父類時,兩個父類中有相同的功能,那么子類對象調(diào)用該功能時,運(yùn)行哪一個呢?因為父類中的方法中存在方法體。但是java支持多重繼承。A繼承BB繼承CC繼承D。多重繼承的出現(xiàn),就有了繼承體系。體系中的頂層父類是通過不斷向上抽取而來的。它里面定義的該體系最基本最共性內(nèi)容的功能。所以,一個體系要想被使用,直接查閱該系統(tǒng)中的父類的功能即可知道該體系的基本用法。那么想要使用一個體系時,需要建立對象。建議建立最子類對象,因為最子類不僅可以使用父類中的功能。還可以使用子類特有的一些功能。簡單說:對于一個繼承體系的使用,查閱頂層父類中的內(nèi)容,創(chuàng)建最底層子類的對象。子父類出現(xiàn)后,類中的成員都有了哪些特點(diǎn):1:成員變量。當(dāng)子父類中出現(xiàn)一樣的屬性時,子類類型的對象,調(diào)用該屬性,值是子類的屬性值。如果想要調(diào)用父類中的屬性值,需要使用一個關(guān)鍵字:superThis:代表是本類類型的對象引用。Super:代表是子類所屬的父類中的內(nèi)存空間引用。注意:子父類中通常是不會出現(xiàn)同名成員變量的,因為父類中只要定義了,子類就不用在定義了,直接繼承過來用就可以了。2:成員函數(shù)。當(dāng)子父類中出現(xiàn)了一模一樣的方法時,建立子類對象會運(yùn)行子類中的方法。好像父類中的方法被覆蓋掉一樣。所以這種情況,是函數(shù)的另一個特性:覆蓋(復(fù)寫,重寫)什么時候使用覆蓋呢?當(dāng)一個類的功能內(nèi)容需要修改時,可以通過覆蓋來實(shí)現(xiàn)。3:構(gòu)造函數(shù)。發(fā)現(xiàn)子類構(gòu)造函數(shù)運(yùn)行時,先運(yùn)行了父類的構(gòu)造函數(shù)。為什么呢?原因:子類的所有構(gòu)造函數(shù)中的第一行,其實(shí)都有一條隱身的語句super();super():表示父類的構(gòu)造函數(shù),并會調(diào)用于參數(shù)相對應(yīng)的父類中的構(gòu)造函數(shù)。而super():是在調(diào)用父類中空參數(shù)的構(gòu)造函數(shù)。為什么子類對象初始化時,都需要調(diào)用父類中的函數(shù)?(為什么要在子類構(gòu)造函數(shù)的第一行加入這個super()?)因為子類繼承父類,會繼承到父類中的數(shù)據(jù),所以必須要看父類是如何對自己的數(shù)據(jù)進(jìn)行初始化的。所以子類在進(jìn)行對象初始化時,先調(diào)用父類的構(gòu)造函數(shù),這就是子類的實(shí)例化過程。注意:子類中所有的構(gòu)造函數(shù)都會默認(rèn)訪問父類中的空參數(shù)的構(gòu)造函數(shù),因為每一個子類構(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ù)也一樣會訪問父類中的構(gòu)造函數(shù)。問題:super()和this()是否可以同時出現(xiàn)的構(gòu)造函數(shù)中。兩個語句只能有一個定義在第一行,所以只能出現(xiàn)其中一個。super()或者this():為什么一定要定義在第一行?因為super()或者this()都是調(diào)用構(gòu)造函數(shù),構(gòu)造函數(shù)用于初始化,所以初始化的動作要先完成。繼承的細(xì)節(jié):什么時候使用繼承呢?當(dāng)類與類之間存在著所屬關(guān)系時,才具備了繼承的前提。a是b中的一種。a繼承b。狼是犬科中的一種。英文書中,所屬關(guān)系:"isa"注意:不要僅僅為了獲取其他類中的已有成員進(jìn)行繼承。所以判斷所屬關(guān)系,可以簡單看,如果繼承后,被繼承的類中的功能,都可以被該子類所具備,那么繼承成立。如果不是,不可以繼承。細(xì)節(jié)二:在方法覆蓋時,注意兩點(diǎn):1:子類覆蓋父類時,必須要保證,子類方法的權(quán)限必須大于等于父類方法權(quán)限可以實(shí)現(xiàn)繼承。否則,編譯失敗。2:覆蓋時,要么都靜態(tài),要么都不靜態(tài)。(靜態(tài)只能覆蓋靜態(tài),或者被靜態(tài)覆蓋)繼承的一個弊端:打破了封裝性。對于一些類,或者類中功能,是需要被繼承,或者復(fù)寫的。這時如何解決問題呢?介紹一個關(guān)鍵字‘final:最終。final特點(diǎn):1:這個關(guān)鍵字是一個修飾符,可以修飾類,方法,變量。2:被final修飾的類是一個最終類,不可以被繼承。3:被final修飾的方法是一個最終方法,不可以被覆蓋。4:被final修飾的變量是一個常量,只能賦值一次。其實(shí)這樣的原因的就是給一些固定的數(shù)據(jù)起個閱讀性較強(qiáng)的名稱。不加final修飾不是也可以使用嗎?那么這個值是一個變量,是可以更改的。加了final,程序更為嚴(yán)謹(jǐn)。常量名稱定義時,有規(guī)范,所有字母都大寫,如果由多個單詞組成,中間用_連接。抽象類:abstract抽象:不具體,看不明白。抽象類表象體現(xiàn)。在不斷抽取過程中,將共性內(nèi)容中的方法聲明抽取,但是方法不一樣,沒有抽取,這時抽取到的方法,并不具體,需要被指定關(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í)例化。否則,該子類還是一個抽象類。抽象類的細(xì)節(jié):1:抽象類中是否有構(gòu)造函數(shù)?有,用于給子類對象進(jìn)行初始化。2:抽象類中是否可以定義非抽象方法?可以。其實(shí),抽象類和一般類沒有太大的區(qū)別,都是在描述事物,只不過抽象類在描述事物時,有些功能不具體。所以抽象類和一般類在定義上,都是需要定義屬性和行為的。只不過,比一般類多了一個抽象函數(shù)。而且比一般類少了一個創(chuàng)建對象的部分。3:抽象關(guān)鍵字abstract和哪些不可以共存?final,private,static4:抽象類中可不可以不定義抽象方法?可以。抽象方法目的僅僅為了不讓該類創(chuàng)建對象。模板方法設(shè)計模式:解決的問題:當(dāng)功能內(nèi)部一部分實(shí)現(xiàn)時確定,一部分實(shí)現(xiàn)是不確定的。這時可以把不確定的部分暴露出去,讓子類去實(shí)現(xiàn)。abstractclassGetTime{publicfinalvoidgetTime(){〃此功能如果不需要復(fù)寫,可加fina1限定longstart=System.currentTimeMillis();code();//不確定的功能部分,提取出來,通過抽象方法實(shí)現(xiàn)longend=System.currentTimeMillis();System.out.println("毫秒是:"+(end-start));}publicabstractvoidcode();//抽象不確定的功能,讓子類復(fù)寫實(shí)現(xiàn)}classSubDemoextendsGetTime{publicvoidcode(){〃子類復(fù)寫功能方法for(inty=0;y<1000;y++){System.out.println("y");}}}接口:★★★★★1:是用關(guān)鍵字interface定義的。2:接口中包含的成員,最常見的有全局常量、抽象方法。注意:接口中的成員都有固定的修飾符。成員變量:publicstaticfinal成員方法:publicabstractinterfaceInter{publicstaticfinalintx=3;publicabstractvoidshow();}3:接口中有抽象方法,說明接口不可以實(shí)例化。接口的子類必須實(shí)現(xiàn)了接口中所有的抽象方法后,該子類才可以實(shí)例化。否則,該子類還是一個抽象類。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:一個類在繼承另一個類的同時,還可以實(shí)現(xiàn)多個接口。所以接口的出現(xiàn)避免了單繼承的局限性。還可以將類進(jìn)行功能的擴(kuò)展。7:其實(shí)java中是有多繼承的。接口與接口之間存在著繼承關(guān)系,接口可以多繼承接口接口都用于設(shè)計上,設(shè)計上的特點(diǎn)(可以理解主板上提供的接口)1:接口是對外提供的規(guī)則。2:接口是功能的擴(kuò)展。3:接口的出現(xiàn)降低了耦合性。抽象類與接口:抽象類:一般用于描述一個體系單元,將一組共性內(nèi)容進(jìn)行抽取,特點(diǎn):可以在類中定義抽象內(nèi)容讓子類實(shí)現(xiàn)可以定義非抽象內(nèi)容讓子類直接使用。它里面定義的都是一些體系中的基本內(nèi)容。接口:一般用于定義對象的擴(kuò)展功能,是在繼承之外還需這個對象具備的一些功能。抽象類和接口的共性:都是不斷向上抽取的結(jié)果。抽象類和接口的區(qū)別:1:抽象類只能被繼承,而且只能單繼承。接口需要被實(shí)現(xiàn),而且可以多實(shí)現(xiàn)。2:抽象類中可以定義非抽象方法,子類可以直接繼承使用。接口中都有抽象方法,需要子類去實(shí)現(xiàn)。3:抽象類使用的是isa關(guān)系。接口使用的likea關(guān)系。4:抽象類的成員修飾符可以自定義。接口中的成員修飾符是固定的。全都是public的。在開發(fā)之前,先定義規(guī)則,A和B分別開發(fā),A負(fù)責(zé)實(shí)現(xiàn)這個規(guī)則,B負(fù)責(zé)使用這個規(guī)則。至于A是如何對規(guī)則具體實(shí)現(xiàn)的,B是不需要知道的。這樣這個接口的出現(xiàn)就降低了A和B直接耦合性。

多態(tài)^***%面向?qū)ο筇卣髦欢鄳B(tài)^***%體現(xiàn):父類引用或者接口的引用指向了自己的子類對象。//Animala=newCat();多態(tài)的好處:提高了程序的擴(kuò)展性。多態(tài)的弊端:當(dāng)父類引用指向子類對象時,雖然提高了擴(kuò)展性,但是只能訪問父類中具備的方法,不可以訪問子類中特有的方法。(前期不能使用后期產(chǎn)生的功能,即訪問的局限性)多態(tài)的前提:1:必須要有關(guān)系,比如繼承、或者實(shí)現(xiàn)。2:通常會有覆蓋操作。多態(tài)的出現(xiàn)思想上也做著變化:以前是創(chuàng)建對象并指揮對象做事情。有了多態(tài)以后,我們可以找到對象的共性類型,直接操作共性類型做事情即可,這樣可以指揮一批對象做事情,即通過操作父類或接口實(shí)現(xiàn)。class畢姥爺{void講課(){System.out.println("企業(yè)管理");}void釣魚(){System.out.println("釣魚");}}class畢老師extends畢姥爺{void講課(){System.out.println("JAVA");}void看電影(){System.out.println("看電影");}}class{publicstaticvoidmain(String[]args){畢姥爺x=new畢老師();〃畢老師對象被提升為了畢姥爺類型。//x.講課();//x.看電影();//錯誤.畢老師y=(畢老師)x;〃將畢姥爺類型強(qiáng)制轉(zhuǎn)換成畢老師類型。y.看電影();//在多態(tài)中,自始自終都是子類對象在做著類型的變化。}}如果想用子類對象的特有方法,如何判斷對象是哪個具體的子類類型呢?可以可以通過一個關(guān)鍵字instanceof;//判斷對象是否實(shí)現(xiàn)了指定的接口或繼承了指定的類格式:〈對象instanceof類型〉,判斷一個對象是否所屬于指定的類型。StudentinstanceofPerson=true;//student繼承了person類多態(tài)在子父類中的成員上的體現(xiàn)的特點(diǎn):1,成員變量:在多態(tài)中,子父類成員變量同名。在編譯時期:參考的是引用型變量所屬的類中是否有調(diào)用的成員。(編譯時不產(chǎn)生對象,只檢查語法錯誤)運(yùn)行時期:也是參考引用型變量所屬的類中是否有調(diào)用的成員。簡單一句話:無論編譯和運(yùn)行,成員變量參考的都是引用變量所屬的類中的成員變量。再說的更容易記憶一些:成員變量編譯運(yùn)行都看=左邊。2,成員函數(shù)。編譯時期:參考引用型變量所屬的類中是否有調(diào)用的方法。運(yùn)行事情:參考的是對象所屬的類中是否有調(diào)用的方法。為什么是這樣的呢?因為在子父類中,對于一模一樣的成員函數(shù),有一個特性:覆蓋。簡單一句:成員函數(shù),編譯看引用型變量所屬的類,運(yùn)行看對象所屬的類。更簡單:成員函數(shù)編譯看=左邊,運(yùn)行看=右邊。3,靜態(tài)函數(shù)。編譯時期:參考的是引用型變量所屬的類中是否有調(diào)用的成員。運(yùn)行時期:也是參考引用型變量所屬的類中是否有調(diào)用的成員。為什么是這樣的呢?因為靜態(tài)方法,其實(shí)不所屬于對象,而是所屬于該方法所在的類。調(diào)用靜態(tài)的方法引用是哪個類的引用調(diào)用的就是哪個類中的靜態(tài)方法。簡單說:靜態(tài)函數(shù)編譯運(yùn)行都看=左邊。java.lang.ObjectObjeCt:所有類的直接或者間接父類,Java認(rèn)為所有的對象都具備一些基本的共性內(nèi)容,這些內(nèi)容可以不斷的向上抽取,最終就抽取到了一個最頂層的類中的,該類中定義的就是所有對象都具備的功能。具體方法:booleanequals(Objectobj):用于比較兩個對象是否相等,其實(shí)內(nèi)部比較的就是兩個對象地址。而根據(jù)對象的屬性不同,判斷對象是否相同的具體內(nèi)容也不一樣。所以在定義類時,一般都會復(fù)寫equals方法,建立本類特有的判斷對象是否相同的依據(jù)。publicbooleanequals(Objectobj){if(!(objinstanceofPerson))returnfalse;Personp=(Person)obj;returnthis.age==p.age;}2,StringtoString():將對象變成字符串;默認(rèn)返回的格式:類名@哈希值=getClass().getName()+'@'+Integer.toHexString(hashCode())為了對象對應(yīng)的字符串內(nèi)容有意義,可以通過復(fù)寫,建立該類對象自己特有的字符串表現(xiàn)形式。publicStringtoString(){return"person:"+age;}3,ClassgetClass():獲取任意對象運(yùn)行時的所屬字節(jié)碼文件對象。4,inthashCode():返回該對象的哈希碼值。支持此方法是為了提高哈希表的性能。通常equals,toString,hashCode,在應(yīng)用中都會被復(fù)寫,建立具體對象的特有的內(nèi)容。內(nèi)部類:如果A類需要直接訪問B類中的成員,而B類又需要建立A類的對象。這時,為了方便設(shè)計和訪問,直接將A類定義在B類中。就可以了。A類就稱為內(nèi)部類。內(nèi)部類可以直接訪問外部類中的成員。而外部類想要訪問內(nèi)部類,必須要建立內(nèi)部類的對象。classOuter{intnum=4;classInner{voidshow(){System.out.println("innershowrun"+num);}}publicvoidmethod(){Innerin=newInner();//創(chuàng)建內(nèi)部類的對象。in.show();//調(diào)用內(nèi)部類的方法。}}當(dāng)內(nèi)部類定義在外部類中的成員位置上,可以使用一些成員修飾符修飾private、static。1:默認(rèn)修飾符。直接訪問內(nèi)部類格式:外部類名.內(nèi)部類名變量名=外部類對象.內(nèi)部類對象;Outer.Innerin=newOuter.newInner();//這種形式很少用。但是這種應(yīng)用不多見,因為內(nèi)部類之所以定義在內(nèi)部就是為了封裝。想要獲取內(nèi)部類對象通常都通過外部類的方法來獲取。這樣可以對內(nèi)部類對象進(jìn)行控制。2:私有修飾符。通常內(nèi)部類被封裝,都會被私有化,因為封裝性不讓其他程序直接訪問。3:靜態(tài)修飾符。如果內(nèi)部類被靜態(tài)修飾,相當(dāng)于外部類,會出現(xiàn)訪問局限性,只能訪問外部類中的靜態(tài)成員。注意;如果內(nèi)部類中定義了靜態(tài)成員,那么該內(nèi)部類必須是靜態(tài)的。內(nèi)部類編譯后的文件名為:“外部類名$內(nèi)部類名.java”;為什么內(nèi)部類可以直接訪問外部類中的成員呢?那是因為內(nèi)部中都持有一個外部類的引用。這個是引用是外部類名.this內(nèi)部類可以定義在外部類中的成員位置上,也可以定義在外部類中的局部位置上。當(dāng)內(nèi)部類被定義在局部位置上,只能訪問局部中被final修飾的局部變量。匿名內(nèi)部類:沒有名字的內(nèi)部類。就是內(nèi)部類的簡化形式。一般只用一次就可以用這種形式。匿名內(nèi)部類其實(shí)就是一個匿名子類對象。想要定義匿名內(nèi)部類:需要前提,內(nèi)部類必須繼承一個類或者實(shí)現(xiàn)接口。匿名內(nèi)部類的格式:new父類名&接口名(){定義子類成員或者覆蓋父類方法}.方法。匿名內(nèi)部類的使用場景:當(dāng)函數(shù)的參數(shù)是接口類型引用時,如果接口中的方法不超過3個。可以通過匿名內(nèi)部類來完成參數(shù)的傳遞。其實(shí)就是在創(chuàng)建匿名內(nèi)部類時,該類中的封裝的方法不要過多,最好兩個或者兩個以內(nèi)。//面試//1newObject(){voidshow(){System.out.println("showrun");}}.show();//2Objectobj=newObject(){voidshow(){System.out.println("showrun");}};obj.show();1和2的寫法正確嗎?有區(qū)別嗎?說出原因。寫法是正確,1和2都是在通過匿名內(nèi)部類建立一個Object類的子類對象。區(qū)別:第一個可是編譯通過,并運(yùn)行。第二個編譯失敗,因為匿名內(nèi)部類是一個子類對象,當(dāng)用Object的obj引用指向時,就被提升為了Object類型,而編譯時檢查Object類中是否有show方法,所以編譯失敗。classInnerClassDemo6{+(static)classInner{voidshow(){}}publicvoidmethod(){this.newInner().show();//可以}publicstaticvoidmain(String[]args){//static不允許thisThis.newInner().show();//錯誤,Inner類需要定義成static}}interfaceInter{voidshow();}classOuter{//通過匿名內(nèi)部類補(bǔ)足Outer類中的代碼。publicstaticIntermethod(){returnnewInter(){publicvoidshow(){}};}}classInnerClassDemo7{publicstaticvoidmain(String[]args){Outer.method().show();/*Outer.method():意思是:Outer中有一個名稱為method的方法,而且這個方法是靜態(tài)的。Outer.method().show():當(dāng)Outer類調(diào)用靜態(tài)的method方法運(yùn)算結(jié)束后的結(jié)果又調(diào)用了show方法,意味著:method()方法運(yùn)算完一個是對象,而且這個對象是Inter類型的。*/function(newInter(){publicvoidshow(){}});//匿名內(nèi)部類作為方法的參數(shù)進(jìn)行傳遞。}publicstaticvoidfunction(Interin){in.show();}}異常:★★★★異常:就是不正常。程序在運(yùn)行時出現(xiàn)的不正常情況。其實(shí)就是程序中出現(xiàn)的問題。這個問題按照面向?qū)ο笏枷脒M(jìn)行描述,并封裝成了對象。因為問題的產(chǎn)生有產(chǎn)生的原因、有問題的名稱、有問題的描述等多個屬性信息存在。當(dāng)出現(xiàn)多屬性信息最方便的方式就是將這些信息進(jìn)行封裝。異常就是java按照面向?qū)ο蟮乃枷雽栴}進(jìn)行對象封裝。這樣就方便于操作問題以及處理問題。出現(xiàn)的問題有很多種,比如角標(biāo)越界,空指針等都是。就對這些問題進(jìn)行分類。而且這些問題都有共性內(nèi)容比如:每一個問題都有名稱,同時還有問題描述的信息,問題出現(xiàn)的位置,所以可以不斷的向上抽取。形成了異常體系。java.lang.Throwable:Throwable:可拋出的?!?-Error:錯誤,一般情況下,不編寫針對性的代碼進(jìn)行處理,通常是jvm發(fā)生的,需要對程序進(jìn)行修正。|--Exception:異常,可以有針對性的處理方式無論是錯誤還是異常,它們都有具體的子類體現(xiàn)每一個問題,它們的子類都有一個共性,就是都以父類名才作為子類的后綴名。這個體系中的所有類和對象都具備一個獨(dú)有的特點(diǎn);就是可拋性??蓲佇缘捏w現(xiàn):就是這個體系中的類和對象都可以被throws和throw兩個關(guān)鍵字所操作。classExceptionDemo{publicstaticvoidmain(String[]args){//byte[]buf=newbyte[1024*1024*700];//java.lang.0utOfMemoryError內(nèi)存溢出錯誤}}在開發(fā)時,如果定義功能時,發(fā)現(xiàn)該功能會出現(xiàn)一些問題,應(yīng)該將問題在定義功能時標(biāo)示出來,這樣調(diào)用者就可以在使用這個功能的時候,預(yù)先給出處理方式。如何標(biāo)示呢?通過throws關(guān)鍵字完成,格式:throws異常類名,異常類名...這樣標(biāo)示后,調(diào)用者,在使用該功能時,就必須要處理,否則編譯失敗。處理方式有兩種:1、捕捉;2、拋出。對于捕捉:java有針對性的語句塊進(jìn)行處理。try{需要被檢測的代碼;}catch(異常類變量名){異常處理代碼;}fianlly{一定會執(zhí)行的代碼;}catch(Exceptione){//e用于接收try檢測到的異常對象。System.out.println("message:"+e.getMessage());//獲取的是異常的信息。System.out.println("toString:"+e.toString());//獲取的是異常的名字+異常的信息。e.printStackTrace();//打印異常在堆棧中信息;異常名稱+異常信息+異常的位置。}異常處理原則:功能拋出幾個異常,功能調(diào)用如果進(jìn)行try處理,需要與之對應(yīng)的catch處理代碼塊,這樣的處理有針對性,拋幾個就處理幾個。特殊情況:try對應(yīng)多個catch時,如果有父類的catch語句塊,一定要放在下面。throw和throws關(guān)鍵字的區(qū)別:throw用于拋出異常對象,后面跟的是異常對象;throw用在函數(shù)內(nèi)。throws用于拋出異常類,后面跟的異常類名,可以跟多個,用逗號隔開。throws用在函數(shù)上。通常情況:函數(shù)內(nèi)容如果有throw,拋出異常對象,并沒有進(jìn)行處理,那么函數(shù)上一定要聲明,否則編譯失敗。但是也有特殊情況。異常分兩種:1:編譯時被檢查的異常,只要是Exception及其子類都是編譯時被檢測的異常。2:運(yùn)行時異常,其中Exception有一個特殊的子類RuntimeException,以及RuntimeException的子類是運(yùn)行異常,也就說這個異常是編譯時不被檢查的異常。編譯時被檢查的異常和運(yùn)行時異常的區(qū)別:編譯被檢查的異常在函數(shù)內(nèi)被拋出,函數(shù)必須要聲明,否編譯失敗。聲明的原因:是需要調(diào)用者對該異常進(jìn)行處理。運(yùn)行時異常如果在函數(shù)內(nèi)被拋出,在函數(shù)上不需要聲明。不聲明的原因:不需要調(diào)用者處理,運(yùn)行時異常發(fā)生,已經(jīng)無法再讓程序繼續(xù)運(yùn)行,所以,不讓調(diào)用處理的直接讓程序停止,由調(diào)用者對代碼進(jìn)行修正。定義異常處理時,什么時候定義try,什么時候定義throws呢?功能內(nèi)部如果出現(xiàn)異常,如果內(nèi)部可以處理,就用try;如果功能內(nèi)部處理不了,就必須聲明出來,讓調(diào)用者處理。自定義異常:當(dāng)開發(fā)時,項目中出現(xiàn)了java中沒有定義過的問題時,這時就需要我們按照java異常建立思想,將項目的中的特有問題也進(jìn)行對象的封裝。這個異常,稱為自定義異常。

對于除法運(yùn)算,0作為除數(shù)是不可以的。java中對這種問題用ArithmeticException類進(jìn)行描述。對于這個功能,在我們項目中,除數(shù)除了不可以為0外,還不可以為負(fù)數(shù)??墒秦?fù)數(shù)的部分java并沒有針對描述。所以我們就需要自定義這個異常。自定義異常的步驟:1:定義一個子類繼承Exception或RuntimeException,讓該類具備可拋性。2:通過throw或者throws進(jìn)行操作。異常的轉(zhuǎn)換思想:當(dāng)出現(xiàn)的異常是調(diào)用者處理不了的,就需要將此異常轉(zhuǎn)換為一個調(diào)用者可以處理的異常拋出3,tryfinallytrycatchfinally的幾種結(jié)合方式:3,tryfinally1,2,trytrycatchcatchfinally這種情況,如果出現(xiàn)異常,并不處理,但是資源一定關(guān)閉,所以tryfinally集合只為關(guān)閉資源記住:finally很有用,主要用戶關(guān)閉資源。無論是否發(fā)生異常,資源都必須進(jìn)行關(guān)閉。System.exit(0);//退出jvm,只有這種情況finally不執(zhí)行。當(dāng)異常出現(xiàn)后,在子父類進(jìn)行覆蓋時,有了一些新的特點(diǎn):1:當(dāng)子類覆蓋父類的方法時,如果父類的方法拋出了異常,那么子類的方法要么不拋出異常要么拋出父類異常或者該異常的子類,不能拋出其他異常。2:如果父類拋出了多個異常,那么子類在覆蓋時只能拋出父類的異常的子集。V-注意:如果父類或者接口中的方法沒有拋出過異常,那么子類是不可以拋出異常的,如果子類的覆蓋的方法中出現(xiàn)了異常,只能try不能throws。如果這個異常子類無法處理,已經(jīng)影響了子類方法的具體運(yùn)算,這時可以在子類方法中,通過throw拋出RuntimeException異?;蛘咂渥宇?,這樣,子類的方法上是不需要throws聲明的。常見異常:1、腳標(biāo)越界異常(IndexOutOfBoundsException)包括數(shù)組、字符串;空指針異常(NullPointerException)2、類型轉(zhuǎn)換異常:ClassCastException3、沒有這個元素異常:NullPointerException4、不支持操作異常;異常要盡量避免,如果避免不了,需要預(yù)先給出處理方式。比如家庭備藥,比如滅火器。包:定義包用package關(guān)鍵字。1:對類文件進(jìn)行分類管理。2:給類文件提供多層名稱空間。如果生成的包不在當(dāng)前目錄下,需要最好執(zhí)行classpath,將包所在父目錄定義到classpath變量中即可。一般在定義包名時,因為包的出現(xiàn)是為了區(qū)分重名的類。所以包名要盡量唯一。怎么保證唯一性呢?可以使用url域名來進(jìn)行包名稱的定義。packagepack;//定義了一個包,名稱為pack。注意:包名的寫法規(guī)范:所有字母都小寫。//packagecn.itcast.pack.demo;類的全名稱是包名.類名編譯命令:javac-d位置(?當(dāng)前路徑)java源文件(就可以自動生成包)包是一種封裝形式,用于封裝類,想要被包以外的程序訪問,該類必須public;類中的成員,如果被包以外訪問,也必須public;包與包之間訪問可以使用的權(quán)限有兩種:1:public2:protected:只能是不同包中的子類可以使用的權(quán)限??偨Y(jié)java中的四種權(quán)限:范圍publicprotecteddefaultprivate同一個類中okokokok同一包中okokok子類ok不同包中okImport-導(dǎo)入:類名稱變長,寫起來很麻煩。為了簡化,使用了一個關(guān)鍵字:import,可以使用這個關(guān)鍵字導(dǎo)入指定包中的類。記?。簩?shí)際開發(fā)時,到的哪個類就導(dǎo)入哪個類,不建議使用*.importpacka.*;//這個僅僅是導(dǎo)入了packa當(dāng)前目錄下的所有的類。不包含子包。importpacka.abc.*;//導(dǎo)入了packa包中的子包abc下的當(dāng)前的所有類。如果導(dǎo)入的兩個包中存在著相同名稱的類。這時如果用到該類,必須在代碼中指定包名。常見的軟件包:java.lang:languagejava的核心包,ObjectSystemStringThrowablejdk1.2版本后,該包中的類自動被導(dǎo)入。java.awt:定義的都是用于java圖形界面開發(fā)的對象。javax.swing:提供所有的windows桌面應(yīng)用程序包括的控件,比如:Frame,Dialog,Table,List等等,就是java的圖形界面庫。:用于java網(wǎng)絡(luò)編程方面的對象都在該包中。java.io:inputoutput用于操作設(shè)備上數(shù)據(jù)的對象都在該包中。比如:讀取硬盤數(shù)據(jù),往硬盤寫入數(shù)據(jù)。java.util:java的工具包,時間對象,集合框架。java.applet:application+let客戶端java小程序。server+let一>servlet服務(wù)端java小程序。jar:java的壓縮包,主要用于存儲類文件,或者配置文件等。命令格式:jar-cf包名.jar包目錄解壓縮:jar-xvf包名.jar將jar包目錄列表重定向到一個文件中:jar-tf包名.jar>c:\1.txt多線程:★★★★進(jìn)程:正在進(jìn)行中的程序。其實(shí)進(jìn)程就是一個應(yīng)用程序運(yùn)行時的內(nèi)存分配空間。線程:其實(shí)就是進(jìn)程中一個程序執(zhí)行控制單元,一條執(zhí)行路徑。進(jìn)程負(fù)責(zé)的是應(yīng)用程序的空間的標(biāo)示。線程負(fù)責(zé)的是應(yīng)用程序的執(zhí)行順序。一個進(jìn)程至少有一個線程在運(yùn)行,當(dāng)一個進(jìn)程中出現(xiàn)多個線程時,就稱這個應(yīng)用程序是多線程應(yīng)用程序,每個線程在棧區(qū)中都有自己的執(zhí)行空間,自己的方法區(qū)、自己的變量。jvm在啟動的時,首先有一個主線程,負(fù)責(zé)程序的執(zhí)行,調(diào)用的是main函數(shù)。主線程執(zhí)行的代碼都在main方法中。當(dāng)產(chǎn)生垃圾時,收垃圾的動作,是不需要主線程來完成,因為這樣,會出現(xiàn)主線程中的代碼執(zhí)行會停止,會去運(yùn)行垃圾回收器代碼,效率較低,所以由單獨(dú)一個線程來負(fù)責(zé)垃圾回收。隨機(jī)性的原理:因為cpu的快速切換造成,哪個線程獲取到j(luò)cpu的執(zhí)行權(quán),哪個線程就執(zhí)行。返回當(dāng)前線程的名稱:Thread.currentThread().getName()線程的名稱是由:Thread-編號定義的。編號從0開始。線程要運(yùn)行的代碼都統(tǒng)一存放在了run方法中。線程要運(yùn)行必須要通過類中指定的方法開啟。start方法。(啟動后,就多了一條執(zhí)行路徑)start方法:1)、啟動了線程;2)、讓jvm調(diào)用了run方法。創(chuàng)建線程的第一種方式:繼承Thread,由子類復(fù)寫run方法。步驟:1,定義類繼承Thread類;2,目的是復(fù)寫run方法,將要讓線程運(yùn)行的代碼都存儲到run方法中;3,通過創(chuàng)建Thread類的子類對象,創(chuàng)建線程對象;4,調(diào)用線程的start方法,開啟線程,并執(zhí)行run方法。線程狀態(tài):被創(chuàng)建:start()運(yùn)行:具備執(zhí)行資格,同時具備執(zhí)行權(quán);凍結(jié):sleep(time),wait()—notify()喚醒;線程釋放了執(zhí)行權(quán),同時釋放執(zhí)行資格;臨時阻塞狀態(tài):線程具備epu的執(zhí)行資格,沒有cpu的執(zhí)行權(quán);消亡:stop()創(chuàng)建線程的第二種方式:實(shí)現(xiàn)一個接口Runnable。步驟:定義類實(shí)現(xiàn)Runnable接口。覆蓋接口中的run方法(用于封裝線程要運(yùn)行的代碼)。通過Thread類創(chuàng)建線程對象;將實(shí)現(xiàn)了Runnable接口的子類對象作為實(shí)際參數(shù)傳遞給Thread類中的構(gòu)造函數(shù)。為什么要傳遞呢?因為要讓線程對象明確要運(yùn)行的run方法所屬的對象。調(diào)用Thread對象的start方法。開啟線程,并運(yùn)行Runnable接口子類中的run方法。Tickett=newTicketO;/*直接創(chuàng)建Ticket對象,并不是創(chuàng)建線程對象。因為創(chuàng)建對象只能通過newThread類,或者newThread類的子類才可以。所以最終想要創(chuàng)建線程。既然沒有了Thread類的子類,就只能用Thread類。*/Threadt1=newThread(t);//創(chuàng)建線程。/*只要將t作為Thread類的構(gòu)造函數(shù)的實(shí)際參數(shù)傳入即可完成線程對象和t之間的關(guān)聯(lián)為什么要將t傳給Thread類的構(gòu)造函數(shù)呢?其實(shí)就是為了明確線程要運(yùn)行的代碼run方法。*/t1.start();為什么要有Runnable接口的出現(xiàn)?1:通過繼承Thread類的方式,可以完成多線程的建立。但是這種方式有一個局限性,如果一個類已經(jīng)有了自己的父類,就不可以繼承Thread類,因為java單繼承的局限性??墒窃擃愔械倪€有部分代碼需要被多個線程同時執(zhí)行。這時怎么辦呢?只有對該類進(jìn)行額外的功能擴(kuò)展,java就提供了一個接口Runnable。這個接口中定義了run方法,其實(shí)run方法的定義就是為了存儲多線程要運(yùn)行的代碼。所以,通常創(chuàng)建線程都用第二種方式。因為實(shí)現(xiàn)Runnable接口可以避免單繼承的局限性。2:其實(shí)是將不同類中需要被多線程執(zhí)行的代碼進(jìn)行抽取。將多線程要運(yùn)行的代碼的位置單獨(dú)定義到接口中。為其他類進(jìn)行功能擴(kuò)展提供了前提。所以Thread類在描述線程時,內(nèi)部定義的run方法,也來自于Runnable接口。實(shí)現(xiàn)Runnable接口可以避免單繼承的局限性。而且,繼承Thread,是可以對Thread類中的方法,進(jìn)行子類復(fù)寫的。但是不需要做這個復(fù)寫動作的話,只為定義線程代碼存放位置,實(shí)現(xiàn)Runnable接口更方便一些。所以Runnable接口將線程要執(zhí)行的任務(wù)封裝成了對象。//面試newThread(newRunnable(){//匿名publicvoidrun(){System.out.println("runnablerun");}}){publicvoidrun(){System.out.println("subthreadrun");}}.start();//結(jié)果:subthreadrunTry{Thread.sleep(10);}catch(InterruptedExceptione){}//當(dāng)刻意讓線程稍微停一下,模擬cpu切換情況。多線程安全問題的原因:通過圖解:發(fā)現(xiàn)一個線程在執(zhí)行多條語句時,并運(yùn)算同一個數(shù)據(jù)時,在執(zhí)行過程中,其他線程參與進(jìn)來,并操作了這個數(shù)據(jù)。導(dǎo)致到了錯誤數(shù)據(jù)的產(chǎn)生。涉及到兩個因素:1,多個線程在操作共享數(shù)據(jù)。2,有多條語句對共享數(shù)據(jù)進(jìn)行運(yùn)算。原因:這多條語句,在某一個時刻被一個線程執(zhí)行時,還沒有執(zhí)行完,就被其他線程執(zhí)行了。解決安全問題的原理:只要將操作共享數(shù)據(jù)的語句在某一時段讓一個線程執(zhí)行完,在執(zhí)行過程中,其他線程不能進(jìn)來執(zhí)行就可以解決這個問題。如何進(jìn)行多句操作共享數(shù)據(jù)代碼的封裝呢?java中提供了一個解決方式:就是同步代碼塊。格式:synchronized(對象){//任意對象都可以。這個對象就是鎖。需要被同步的代碼;}同步:★★★★★好處:解決了線程安全問題。弊端:相對降低性能,因為判斷鎖需要消耗資源,產(chǎn)生了死鎖。定義同步是有前提的:1,必須要有兩個或者兩個以上的線程,才需要同步。2,多個線程必須保證使用的是同一個鎖。同步的第二種表現(xiàn)形式:同步函數(shù):其實(shí)就是將同步關(guān)鍵字定義在函數(shù)上,讓函數(shù)具備了同步性。同步函數(shù)是用的哪個鎖呢?通過驗證,函數(shù)都有自己所屬的對象this,所以同步函數(shù)所使用的鎖就是this鎖。當(dāng)同步函數(shù)被static修飾時,這時的同步用的是哪個鎖呢?靜態(tài)函數(shù)在加載時所屬于類,這時有可能還沒有該類產(chǎn)生的對象,但是該類的字節(jié)碼文件加載進(jìn)內(nèi)存就已經(jīng)被封裝成了對象,這個對象就是該類的字節(jié)碼文件對象。所以靜態(tài)加載時,只有一個對象存在,那么靜態(tài)同步函數(shù)就使用的這個對象。這個對象就是類名.class同步代碼塊和同步函數(shù)的區(qū)別?同步代碼塊使用的鎖可以是任意對象。同步函數(shù)使用的鎖是this,靜態(tài)同步函數(shù)的鎖是該類的字節(jié)碼文件對象。在一個類中只有一個同步,可以使用同步函數(shù)。如果有多同步,必須使用同步代碼塊,來確定不同的鎖所以同步代碼塊相對靈活一些?!锟键c(diǎn)問題:請寫一個延遲加載的單例模式?寫懶漢式;當(dāng)出現(xiàn)多線程訪問時怎么解決?加同步,解決安全問題;效率高嗎?不高;怎樣解決?通過雙重判斷的形式解決。//懶漢式:延遲加載方式。當(dāng)多線程訪問懶漢式時,因為懶漢式的方法內(nèi)對共性數(shù)據(jù)進(jìn)行多條語句的操作。所以容易出現(xiàn)線程安全問題為了解決,加入同步機(jī)制,解決安全問題。但是卻帶來了效率降低。為了效率問題,通過雙重判斷的形式解決。classSingle{privatestaticSingles=null;privateSingle(){}publicstaticSinglegetInstance(){//鎖是誰?字節(jié)碼文件對象;if(s==null){synchronized(Single.class){if(s==null)s=newSingle();}}returns;}}同步死鎖:通常只要將同步進(jìn)行嵌套,就可以看到現(xiàn)象。同步函數(shù)中有同步代碼塊,同步代碼塊中還有同步函數(shù)。線程間通信:思路:多個線

溫馨提示

  • 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
  • 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
  • 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會有圖紙預(yù)覽,若沒有圖紙預(yù)覽就沒有圖紙。
  • 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
  • 5. 人人文庫網(wǎng)僅提供信息存儲空間,僅對用戶上傳內(nèi)容的表現(xiàn)方式做保護(hù)處理,對用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對任何下載內(nèi)容負(fù)責(zé)。
  • 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請與我們聯(lián)系,我們立即糾正。
  • 7. 本站不保證下載資源的準(zhǔn)確性、安全性和完整性, 同時也不承擔(dān)用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。

評論

0/150

提交評論