1.import static是Java 5增加的功能,就是將Import類中的靜態(tài)方法每庆,可以作為本類的靜態(tài)方法來用杂曲。
2.String和StringBuffer他們都可以存儲和操作字符串难菌,即包含多個字符的字符串數(shù)據(jù)。
String類是字符串常量闷游,是不可更改的常量殉了。而StringBuffer是字符串變量,它的對象是可以擴充和修改的播赁。
3.break:跳出語句塊,執(zhí)行下面的語句;continue:跳出當前循環(huán),不執(zhí)行循環(huán)中continue下面的所有語句,開始下一次循環(huán)
4.(1)颂郎、一個try可對應多個catch。
(2)容为、有try必須至少有一個catch或者finally乓序。
(3)、finally塊不是必須的坎背,可有可無替劈。
(4)、一般情況下得滤,當異常發(fā)生時陨献,會執(zhí)行catch塊中的語句,特殊情況:當main方法中拋出異常時懂更,如果程序聲明了該? ? ? 異常處理器眨业,則執(zhí)行相應的catch塊中的語句,如果程序沒有聲明相應的異常處理器膜蛔,則不執(zhí)行catch塊中的語句? ? ? 坛猪,直接拋出異常,異常被直接丟給JVM,而JVM的處理方式就是:直接中斷你的程序.
5.類的初始化順序:
1). 父類static(順序按照代碼書寫順序進行)
2). 子類static(順序按照代碼書寫順序進行)
3). 父類非static域皂股、構(gòu)造器(順序按照代碼書寫順序進行)
4). 子類非static域墅茉、構(gòu)造器(順序按照代碼書寫順序進行)
6.靜態(tài)變量和方法不依賴對象實例,由類名直接調(diào)用呜呐。
JAVA的面向?qū)ο缶幊?/b>
面向?qū)ο笾饕槍γ嫦蜻^程就斤。
面向過程的基本單元是函數(shù)。
什么是對象:EVERYTHING IS OBJECT(萬物皆對象)
所有的事物都有兩個方面:
有什么(屬性):用來描述對象蘑辑。
能夠做什么(方法):告訴外界對象有那些功能洋机。
后者以前者為基礎。
大的對象的屬性也可以是一個對象洋魂。
為什么要使用面向?qū)ο螅?/b>
首先绷旗,面向?qū)ο蠓先祟惪创挛锏囊话阋?guī)律。
對象的方法的實現(xiàn)細節(jié)是屏蔽的副砍,只有對象方法的實現(xiàn)者了解細節(jié)衔肢。
方法的定義非常重要。方法有參數(shù)豁翎,也可能有返回值角骤。
注意區(qū)分:對象(本身)、對象的實現(xiàn)者心剥、對象的調(diào)用者邦尊。
分析對象主要從方法開始背桐。
我們通過類來看待對象,類是對象的抽象蝉揍。
其次链峭,采用面向?qū)ο蠓椒梢允瓜到y(tǒng)各部分各司其職、各盡所能疑苫。
對象之間的耦合性一定要低(比如不同硬盤和不同主板之間的關系)熏版。這樣才能使每個對象本身做成最好的纷责。
對于對象的要求:高內(nèi)聚捍掺、低耦合,這樣容易拼裝成為一個系統(tǒng)再膳。
實現(xiàn)高內(nèi)聚就是要最大限度低提高復用性(復用性好是因為高內(nèi)聚)挺勿。
可復用性是OOP的基礎。
比較面向過程的思想和面向?qū)ο蟮乃枷耄?/b>
面向過程的思想:由過程喂柒、步驟不瓶、函數(shù)組成,以過程為核心灾杰;
面向?qū)ο蟮乃枷耄阂詫ο鬄橹行奈秘ぃ乳_發(fā)類,得到對象艳吠,通過對象之間相互通信實現(xiàn)功能麦备。
面向過程是先有算法,后有數(shù)據(jù)結(jié)構(gòu)昭娩。
面向?qū)ο笫窍扔袛?shù)據(jù)結(jié)構(gòu)凛篙,然后再有算法。
在用面向?qū)ο笏枷腴_發(fā)的過程中栏渺,可以復用對象就進行復用呛梆,如無法進行復用則開發(fā)新的對象。
開發(fā)過程是用對個簡單的對象的多個簡單的方法,來實現(xiàn)復雜的功能撮慨。
從語法上來看岸军,一個類是一個新的數(shù)據(jù)類型。
在面向?qū)ο缶幊讨兄突牵撕唵螖?shù)據(jù)類型,就是對象類型神僵。
定義類的格式:
classStudent{
代碼
}
注意類名中單詞的首字母大寫雁刷。
實例變量:定義在類中但在任何方法之外。(New出來的均有初值)
局部變量:定義在方法之中的變量保礼。
局部變量要先賦值沛励,再進行運算责语,而實例變量均已經(jīng)賦初值。這是局部變量和實例變量的一大區(qū)別目派。
實例變量的對象賦值為null坤候。
局部變量不允許范圍內(nèi)定義兩個同名變量。實例變量的作用域在本類中完全有效企蹭,當被其他的類調(diào)用的時候也可能有效白筹。
實例變量和局部變量允許命名沖突。
書寫方法的格式:
修飾符返回值方法名調(diào)用過程中方法體
可能出現(xiàn)的例外
publicint/voidaddNumber(參數(shù))throw Excepion{}
例:
publicintaddNumber(int a,int b){
}
注:方法名中的參數(shù)int a,int b為局部變量
類方法中的一類特殊方法:構(gòu)造方法谅摄。
構(gòu)造方法是當用類生成對象時徒河,系統(tǒng)在生成對象的過程中利用的方法。
注意:構(gòu)造方法在生成對象的時候會被調(diào)用送漠,但并不是構(gòu)造方法生成了對象顽照。
構(gòu)造方法沒有返回值。格式為:public方法名闽寡。
構(gòu)造方法的方法名與類名相同代兵。
構(gòu)造方法是在對象生成的過程中自動調(diào)用,不可能利用指令去調(diào)用爷狈。
在一個對象的生成周期中構(gòu)造方法只用一次植影,一旦這個對象生成,那么這個構(gòu)造方法失效涎永。
用類來生成對象的語句:
Students=new Student()思币。
第一個Student表示這是用Student類進行定義。“Student()”表示調(diào)用一個無參數(shù)的構(gòu)造方法土辩。
如果()中有參數(shù)支救,則系統(tǒng)構(gòu)造對象的過程中調(diào)用有參的方法。
此時S稱為一個對象變量拷淘。
Students的存儲區(qū)域存放的是地址:一個對象在硬盤上占有一個連續(xù)地址各墨,首地址賦予s空間。
S稱為對象Student的引用启涯。
注意:在對象變量中存放的是引用(地址)贬堵;在簡單變量中存放的是數(shù)值。
可以構(gòu)造多個構(gòu)造方法结洼,但多個構(gòu)造方法的參數(shù)表一定不同黎做,參數(shù)順序不同即屬于不同的構(gòu)造方法:
public student(string name,int a){
}
public student(int a,string name){
}
為兩個不同的構(gòu)造方法。
如果我們未給系統(tǒng)提供一個構(gòu)造方法松忍,那么系統(tǒng)會自動提供一個為空的構(gòu)造方法蒸殿。
練習:寫一個類,定義一個對象,定義兩個構(gòu)造方法:一個有參宏所,一個無參酥艳。
(編寫一個程序驗證對象的傳遞的值為地址)
注意下面這種形式:
static void changename(student stu){stu.setName “LUCY”}
注意生成新的對象與舊對象指向無關,生成新對象生命消亡與舊對象無關爬骤。
面向?qū)ο蠓椒ǖ闹剌d(overloading)和覆蓋(overriding)充石。
在有些JAVA書籍中將overriding稱為重載,overloading稱為過載霞玄。
Overloading在一個類中可以定義多個同名方法骤铃,各個方法的參數(shù)表一定不同。但修飾詞可能相同坷剧,返回值也可能相同惰爬。
在程序的編譯過程中根據(jù)變量類型來找相應的方法。因此也有人認為overloading是編譯時的多態(tài)听隐,以后我們還會學到運行時多態(tài)补鼻。
為什么會存在overloading技術呢哄啄?作為應對方法的細節(jié)雅任。
利用類型的差異來影響對方法的調(diào)用。
吃()可以分為吃肉咨跌,吃菜沪么,吃藥,在一個類中可以定義多個吃方法锌半。
構(gòu)造方法也可以實現(xiàn)overloading禽车。例:
publicvoidteach(){};
publicvoidteach(int a){};
publicvoidteach(String a){}為三種不同的方法。
Overloading方法是從低向高轉(zhuǎn)刊殉。
Byte—short—float—int—long—double殉摔。
在構(gòu)造方法中,this表示本類的其他構(gòu)造方法:
student(){};
student(string n){
this();//表示調(diào)用student()
}
如果調(diào)用student(int a)則為this(int a)记焊。
特別注意:用this調(diào)用其他構(gòu)造方法時逸月,this必須為第一條語句,然后才是其他語句遍膜。
This表示當前對象碗硬。
PublicvoidprintNum(){
Int number=40;
System.out.println(this.number);
}
此時打印的是實例變量瓢颅,而非局部變量恩尾,即定義在類中而非方法中的變量。
This.number表示實例變量挽懦。
誰調(diào)用this.number那么誰即為當前(this)對象的number方法翰意。
封裝:使對象的屬性盡可能私有,對象的方法盡可能的公開。用private表示此成員屬性為該類的私有屬性冀偶。
Public表示該屬性(方法)公開虎囚;
Private表示該屬性(方法)為只有本類內(nèi)部可以訪問(類內(nèi)部可見)。
(想用private還要用set和get方法供其他方法調(diào)用蔫磨,這樣可以保證對屬性的訪問方式統(tǒng)一淘讥,并且便于維護訪問權限以及屬性數(shù)據(jù)合法性)
如果沒有特殊情況,屬性一定私有堤如,方法該公開的公開蒲列。
如果不指明誰調(diào)用方法,則默認為this搀罢。
區(qū)分實例變量和局部變量時一定要寫this蝗岖。
11.29
繼承:
父類(SuperClass)和子類(SonClass)。
父類的非私有化屬性和方法可以默認繼承到子類榔至。
Class Son extends Father{
}
而如果父類中的私有方法被子類調(diào)用的話抵赢,則編譯報錯。
父類的構(gòu)造方法子類不可以繼承唧取,更不存在覆蓋的問題铅鲤。(非構(gòu)造方法可以)
如果子類訪問父類的構(gòu)造方法,則在編譯的時候提示訪問不到該方法枫弟。
JAVA中不允許多繼承邢享,一個類有且只有一個父類(單繼承)。
JAVA的數(shù)據(jù)結(jié)構(gòu)為樹型結(jié)構(gòu)淡诗,而非網(wǎng)狀骇塘。(JAVA通過接口和內(nèi)部類實現(xiàn)多繼承)
方法的覆蓋(overriding)
方法的重載并不一定是在一個類中:子類可以從父類繼承一個方法,也可以定義一個同名異參的方法韩容,也稱為overloading款违。
當子類從父類繼承一個無參方法,而又定義了一個同樣的無參方法群凶,則子類新寫的方法覆蓋父類的方法插爹,稱為覆蓋。(注意返回值類型也必須相同座掘,否則編譯出錯递惋。)
如果方法不同,則成重載溢陪。
對于方法的修飾詞萍虽,子類方法要比父類的方法范圍更加的寬泛。
父類為public形真,那么子類為private則出現(xiàn)錯誤杉编。
之所以構(gòu)造方法先運行父類再運行子類是因為構(gòu)造方法是無法覆蓋的超全。
以下范圍依次由嚴到寬:
private:本類訪問;
default:表示默認邓馒,不僅本類訪問嘶朱,而且是同包可見。
Protected:同包可見+不同包的子類可見
Public:表示所有的地方均可見光酣。
當構(gòu)造一個對象的時候疏遏,系統(tǒng)先構(gòu)造父類對象,再構(gòu)造子類對象救军。
構(gòu)造一個對象的順序:(注意:構(gòu)造父類對象的時候也是這幾步)
遞歸地構(gòu)造父類對象财异;
順序地調(diào)用本類成員屬性賦初值語句;
本類的構(gòu)造方法唱遭。
Super()表示調(diào)用父類的構(gòu)造方法戳寸。
Super()也和this一樣必須放在第一行。
This()用于調(diào)用本類的構(gòu)造方法拷泽。
如果沒有定義構(gòu)造方法疫鹊,那么就會調(diào)用父類的無參構(gòu)造方法,即super()司致。
要養(yǎng)成良好的編程習慣:就是要加上默認的父類無參的構(gòu)造方法拆吆。
思考:可是如果我們沒有定義無參的構(gòu)造方法,而在程序中構(gòu)造了有參的構(gòu)造方法蚌吸,那么如果方法中沒有參數(shù)锈拨,那么系統(tǒng)還會調(diào)用有參的構(gòu)造方法么?應該不會羹唠。
多態(tài):多態(tài)指的是編譯時類型變化,而運行時類型不變娄昆。
多態(tài)分兩種:
編譯時多態(tài):編譯時動態(tài)重載佩微;
運行時多態(tài):指一個對象可以具有多個類型。
對象是客觀的萌焰,人對對象的認識是主觀的哺眯。
例:
Animal a=new Dog();查看格式名稱扒俯;
Dog d=(Dog)a奶卓。聲明父類來引用子類。
(思考上面的格式)
運行時多態(tài)的三原則:(應用時為覆蓋)
對象不變撼玄;(改變的是主觀認識)
對于對象的調(diào)用只能限于編譯時類型的方法夺姑,如調(diào)用運行時類型方法報錯。
在上面的例子中:Animal a=new Dog()掌猛;對象a的編譯時類型為Animal盏浙,運行時類型為dog。
注意:編譯時類型一定要為運行時類型的父類(或者同類型)。
對于語句:Dog d=(Dog)a废膘。將d強制聲明為a類型竹海,此時d為Dog(),此時d就可以調(diào)用運行時類型丐黄。注意:a和d指向同一對象斋配。
在程序的運行時,動態(tài)類型判定灌闺。運行時調(diào)用運行時類型许起,即它調(diào)用覆蓋后的方法。
關系運算符:instanceof
ainstanceof Animal;(這個式子的結(jié)果是一個布爾表達式)
a為對象變量菩鲜,Animal是類名园细。
上面語句是判定a是否可以貼Animal標簽。如果可以貼則返回true接校,否則返回false猛频。
在上面的題目中:a instanceof Animal返回True,
a instanceof Dog也返回True蛛勉,
instanceof用于判定是否將前面的對象變量賦值后邊的類名鹿寻。
Instanceof一般用于在強制類型轉(zhuǎn)換之前判定變量是否可以強制轉(zhuǎn)換。
如果Animal a=new Animal()诽凌;
Dog d=Dog()a;
此時編譯無誤毡熏,但運行則會報錯。
Animal a=new Dog()相當于下面語句的功能:
Animal a=getAnimal()侣诵;
Public static Animal.getAnimal;
Return new Dog()痢法;
封裝、繼承杜顺、多態(tài)為面向?qū)ο蟮娜蠡ㄌ匦裕?/b>
運行時的動態(tài)類型判定針對的是方法财搁。運行程序訪問的屬性仍為編譯時屬性商架。
Overloading針對的是編譯時類型伺帘,不存在運行時的多態(tài)誉察。
習題:建立一個shape類钠龙,有circle和rect子類蒋困。
Shape類有zhouchang()和area()兩種方法扛芽。
(正方形)squ為rect子類元潘,rect有cha()用于比較長寬的差殿如。
覆蓋時考慮子類的private及父類的public(考慮多態(tài))馁菜,之所以這樣是避免調(diào)用A時出現(xiàn)實際調(diào)用B的情況茴扁。而出現(xiàn)錯誤。
11.29下午講的是教程上的Module6
Module6-7包括:面向?qū)ο蟾呒壔鸬恕?nèi)部類丹弱、集合德撬、反射(暫時不講)、例外躲胳。
面向?qū)ο蟾呒夠押椤⒓虾屠舛际敲嫦驅(qū)ο蟮暮诵膬?nèi)容。
面向?qū)ο蟾呒墸?/b>修飾符:
static:①可修飾變量(屬性)坯苹;②可修飾方法隆檀;③可修飾代碼塊。
Static int data語句說明data為類變量粹湃,為一個類的共享變量恐仑,屬于整個類。
Int data為實例變量为鳄。
例:
static int data;
m1.data=0;
m1.data++的結(jié)果為1,此時m2.data的結(jié)果也為1裳仆。
Static定義的是一塊為整個類共有的一塊存儲區(qū)域,其發(fā)生變化時訪問到的數(shù)據(jù)都時經(jīng)過變化的孤钦。
其變量可以通過類名去訪問:類名.變量名歧斟。與通過訪問對象的編譯時類型訪問類變量為等價的。
Public static voidprintData(){}
表明此類方法為類方法(靜態(tài)方法)
靜態(tài)方法不需要有對象偏形,可以使用類名調(diào)用静袖。
靜態(tài)方法中不允許訪問類的非靜態(tài)成員,包括成員的變量和方法俊扭,因為此時是通過類調(diào)用的队橙,沒有對象的概念。This.data是不可用的萨惑。
一般情況下捐康,主方法是靜態(tài)方法,所以可調(diào)用靜態(tài)方法咒钟,主方法為靜態(tài)方法是因為它是整個軟件系統(tǒng)的入口吹由,而進入入口時系統(tǒng)中沒有任何對象,只能使用類調(diào)用朱嘴。
覆蓋不適用于靜態(tài)方法。
靜態(tài)方法不可被覆蓋粗合。(允許在子類中定義同名靜態(tài)方法萍嬉,但是沒有多態(tài),嚴格的講隙疚,方法間沒有多態(tài)就不能稱為覆蓋)
當static修飾代碼塊時(注:此代碼塊要在此類的任何一個方法之外)壤追,那么這個代碼塊在代碼被裝載進虛擬機生成對象的時候可被裝載一次,以后再也不執(zhí)行了供屉。
一般靜態(tài)代碼塊被用來初始化靜態(tài)成員行冰。
Static通常用于Singleton模式開發(fā):
Singleton是一種設計模式溺蕉,高于語法,可以保證一個類在整個系統(tǒng)中僅有一個對象悼做。
11.30
final可以修飾類疯特、屬性、方法肛走。
當用final修飾類的時候漓雅,此類不可被繼承,即final類沒有子類朽色。這樣可以用final保證用戶調(diào)用時動作的一致性邻吞,可以防止子類覆蓋情況的發(fā)生。
當利用final修飾一個屬性(變量)的時候葫男,此時的屬性成為常量抱冷。
JAVA利用final定義常量(注意在JAVA命名規(guī)范中常量需要全部字母都大寫):
Final int AGE=10;
常量的地址不可改變梢褐,但在地址中保存的值(即對象的屬性)是可以改變的旺遮。
Final可以配合static使用。利职?
Static final int age=10趣效;
在JAVA中利用public static final的組合方式對常量進行標識(固定格式)。
對于在構(gòu)造方法中利用final進行賦值的時候猪贪,此時在構(gòu)造之前系統(tǒng)設置的默認值相對于構(gòu)造方法失效跷敬。
常量(這里的常量指的是實例常量:即成員變量)賦值:
①在初始化的時候通過顯式聲明賦值。Final int x=3热押;
②在構(gòu)造的時候賦值西傀。
局部變量可以隨時賦值。
利用final定義方法:這樣的方法為一個不可覆蓋的方法桶癣。
Public final void print(){}拥褂;
為了保證方法的一致性(即不被改變),可將方法用final定義牙寞。
如果在父類中有final定義的方法饺鹃,那么在子類中繼承同一個方法。
如果一個方法前有修飾詞private或static间雀,則系統(tǒng)會自動在前面加上final悔详。即private和static方法默認均為final方法。
注:final并不涉及繼承惹挟,繼承取決于類的修飾符是否為private茄螃、default、protected還是public连锯。也就是說归苍,是否繼承取決于這個方法對于子類是否可見用狱。
Abstract(抽象)可以修飾類、方法
如果將一個類設置為abstract拼弃,則此類必須被繼承使用夏伊。此類不可生成對象,必須被繼承使用肴敛。
Abstract可以將子類的共性最大限度的抽取出來署海,放在父類中,以提高程序的簡潔性医男。
Abstract雖然不能生成對象砸狞,但是可以聲明,作為編譯時類型镀梭,但不能作為運行時類型刀森。
Final和abstract永遠不會同時出現(xiàn)。
當abstract用于修飾方法時报账,此時該方法為抽象方法研底,此時方法不需要實現(xiàn),實現(xiàn)留給子類覆蓋透罢,子類覆蓋該方法之后方法才能夠生效榜晦。
注意比較:
private void print(){};此語句表示方法的空實現(xiàn)羽圃。
Abstract void print()乾胶;此語句表示方法的抽象,無實現(xiàn)朽寞。
如果一個類中有一個抽象方法识窿,那么這個類一定為一個抽象類。
反之脑融,如果一個類為抽象類喻频,那么其中可能有非抽象的方法。
如果讓一個非抽象類繼承一個含抽象方法的抽象類肘迎,則編譯時會發(fā)生錯誤甥温。因為當一個非抽象類繼承一個抽象方法的時候,本著只有一個類中有一個抽象方法妓布,那么這個類必須為抽象類的原則窿侈。這個類必須為抽象類,這與此類為非抽象沖突秋茫,所以報錯。
所以子類的方法必須覆蓋父類的抽象方法乃秀。方法才能夠起作用肛著。
只有將理論被熟練運用在實際的程序設計的過程中之后圆兵,才能說理論被完全掌握!
為了實現(xiàn)多態(tài)枢贿,那么父類必須有定義殉农。而父類并不實現(xiàn),留給子類去實現(xiàn)局荚。此時可將父類定義成abstract類超凳。如果沒有定義抽象的父類,那么編譯會出現(xiàn)錯誤耀态。
Abstract和static不能放在一起轮傍,否則便會出現(xiàn)錯誤。(這是因為static不可被覆蓋首装,而abstract為了生效必須被覆蓋创夜。)
例:(本例已存在\CODING\abstract\TestClass.java文件中)
public class TestClass{
public static void main(String[] args){
SuperClass sc=new SubClass();
Sc.print();
}
Abstract class SuperClass{
Abstract void print();}
}
class SubClass extends SuperClass(){
void print(){
System.out.println(“print”);}
}
JAVA的核心概念:接口(interface)
接口與類屬于同一層次,實際上仙逻,接口是一種特殊的抽象類驰吓。
如:
interface IA{
}
public interface:公開接口
與類相似,一個文件只能有一個public接口系奉,且與文件名相同檬贰。
在一個文件中不可同時定義一個public接口和一個public類。
一個接口中缺亮,所有方法為公開翁涤、抽象方法;所有的屬性都是公開瞬内、靜態(tài)迷雪、常量。
一個類實現(xiàn)一個接口的格式:
class IAImple implements IA{
};
一個類實現(xiàn)接口虫蝶,相當于它繼承一個抽象類章咧。
類必須實現(xiàn)接口中的方法,否則其為一抽象類能真。
實現(xiàn)中接口和類相同赁严。
接口中可不寫public,但在子類中實現(xiàn)接口的過程中public不可省粉铐。
(如果剩去public則在編譯的時候提示出錯:對象無法從接口中實現(xiàn)方法疼约。)
注:
一個類除繼承另外一個類,還可以實現(xiàn)接口蝙泼;
class IAImplextends java.util.Arrylistimplement IA{}
繼承類實現(xiàn)接口
這樣可以實現(xiàn)變相的多繼承程剥。
一個類只能繼承另外一個類,但是它可以繼承多個接口汤踏,中間用“织鲸,”隔開舔腾。
Implements IA,IB
所謂實現(xiàn)一個接口,就是指實現(xiàn)接口中的方法搂擦。
接口和接口之間可以定義繼承關系稳诚,并且接口之間允許實現(xiàn)多繼承。
例:interface IC extends IA,IB{};
接口也可以用于定義對象
IA I=new IAImpl();
實現(xiàn)的類從父類和接口繼承的都可做運行時類型瀑踢。
IAImple extends A implement IA,IB
IB I=new IAImple();
I instance of IAImple;
I instance of A;
I instance of IA;
I instance of IB;
返回的結(jié)果均為true.
接口和多態(tài)都為JAVA技術的核心扳还。
接口往往被我們定義成一類XX的東西。
接口實際上是定義一個規(guī)范橱夭、標準氨距。
通過接口可以實現(xiàn)不同層次、不同體系對象的共同屬性徘钥;
通過接口實現(xiàn)write once as anywhere.
以JAVA數(shù)據(jù)庫連接為例子:JDBC制定標準衔蹲;數(shù)據(jù)廠商實現(xiàn)標準;用戶使用標準呈础。
接口通常用來屏蔽底層的差異舆驶。
②接口也因為上述原因被用來保持架構(gòu)的穩(wěn)定性。
JAVA中有一個特殊的類:Object而钞。它是JAVA體系中所有類的父類(直接父類或者間接父類)沙廉。
此類中的方法可以使所的類均繼承。
以下介紹的三種方法屬于Object:
finalize方法:當一個對象被垃圾回收的時候調(diào)用的方法臼节。
toString():是利用字符串來表示對象撬陵。
當我們直接打印定義的對象的時候,隱含的是打印toString()的返回值网缝。
可以通過子類作為一個toString()來覆蓋父類的toString()巨税。
以取得我們想得到的表現(xiàn)形式,即當我們想利用一個自定義的方式描述對象的時候粉臊,我們應該覆蓋toString()草添。
(3)equal
首先試比較下例:
String A=new String(“hello”);
String A=new String(“hello”);
A==B(此時程序返回為FALSE)
因為此時AB中存的是地址僵井,因為創(chuàng)建了新的對象剖效,所以存放的是不同的地址。
附加知識:
字符串類為JAVA中的特殊類猛遍,String中為final類屠凶,一個字符串的值不可重復驰后。因此在JAVA VM(虛擬機)中有一個字符串池,專門用來存儲字符串矗愧。如果遇到String a=”hello”時(注意沒有NEW灶芝,不是創(chuàng)建新串),系統(tǒng)在字符串池中尋找是否有”hello”,此時字符串池中沒有”hello”监署,那么系統(tǒng)將此字符串存到字符串池中颤专,然后將”hello”在字符串池中的地址返回a。如果系統(tǒng)再遇到String b=”hello”钠乏,此時系統(tǒng)可以在字符串池中找到“hello”。則會把地址返回b春塌,此時a與b為相同晓避。
String a=”hello”;
System.out.println(a==”hello”);
系統(tǒng)的返回值為true。
故如果要比較兩個字符串是否相同(而不是他們的地址是否相同)只壳∏喂埃可以對a調(diào)用equal:
System.out.println(a.equal(b));
equal用來比較兩個對象中字符串的順序。
a.equal(b)是a與b的值的比較吼句。
注意下面程序:
student a=new student(“LUCY”,20);
student b=new student(“LUCY”,20);
System.out.println(a==b);
System.out.println(a.equal(b));
此時返回的結(jié)果均為false锅必。
以下為定義equal(加上這個定義,返回ture或false)
public boolean equals(Object o){
student s=(student)o;
if (s.name.equals(this.name)&&s.age==this.age)
else return false;
}如果equals()返回的值為
以下為實現(xiàn)標準equals的流程:
public boolean equals(Object o){
if (this==o) return trun;//此時兩者相同
if (o==null) return false;
if (! o instanceof strudent) return false;//不同類
studeng s=(student)o; //強制轉(zhuǎn)換
if (s.name.equals(this.name)&&s.age==this.age) return true;
else return false;
}
以上過程為實現(xiàn)equals的標準過程惕艳。
練習:建立一個employee類搞隐,有String name,int id,double salary.運用get和set方法,使用toString远搪,使用equals劣纲。
封裝類:
JAVA為每一個簡單數(shù)據(jù)類型提供了一個封裝類,使每個簡單數(shù)據(jù)類型可以被Object來裝載谁鳍。
除了int和char癞季,其余類型首字母大寫即成封裝類。
轉(zhuǎn)換字符的方式:
int I=10;
String s=I+” ”;
String s1=String.valueOf(i);
Int I=10;
Interger I_class=new integer(I);
看javadoc的幫助文檔倘潜。
附加內(nèi)容:
“==”在任何時候都是比較地址绷柒,這種比較永遠不會被覆蓋。
程序員自己編寫的類和JDK類是一種合作關系涮因。(因為多態(tài)的存在废睦,可能存在我們調(diào)用JDK類的情況,也可能存在JDK自動調(diào)用我們的類的情況蕊退。)
注意:類型轉(zhuǎn)換中double\interger\string之間的轉(zhuǎn)換最多郊楣。
12.01
內(nèi)部類:
(注:所有使用內(nèi)部類的地方都可以不用內(nèi)部類,使用內(nèi)部類可以使程序更加的簡潔瓤荔,便于命名規(guī)范和劃分層次結(jié)構(gòu))净蚤。
內(nèi)部類是指在一個外部類的內(nèi)部再定義一個類。
內(nèi)部類作為外部類的一個成員输硝,并且依附于外部類而存在的今瀑。
內(nèi)部類可為靜態(tài),可用PROTECTED和PRIVATE修飾。(而外部類不可以:外部類只能使用PUBLIC和DEFAULT)橘荠。
內(nèi)部類的分類:
成員內(nèi)部類屿附、
局部內(nèi)部類、
靜態(tài)內(nèi)部類哥童、
匿名內(nèi)部類(圖形是要用到挺份,必須掌握)。
成員內(nèi)部類:作為外部類的一個成員存在贮懈,與外部類的屬性匀泊、方法并列。
內(nèi)部類和外部類的實例變量可以共存朵你。
在內(nèi)部類中訪問實例變量:this.屬性
在內(nèi)部類訪問外部類的實例變量:外部類名.this.屬性各聘。
成員內(nèi)部類的優(yōu)點:
⑴內(nèi)部類作為外部類的成員,可以訪問外部類的私有成員或?qū)傩月找健#词箤⑼獠款惵暶鳛?/b>PRIVATE躲因,但是對于處于其內(nèi)部的內(nèi)部類還是可見的。)
⑵用內(nèi)部類定義在外部類中不可訪問的屬性忌傻。這樣就在外部類中實現(xiàn)了比外部類的private還要小的訪問權限大脉。
注意:內(nèi)部類是一個編譯時的概念,一旦編譯成功芯勘,就會成為完全不同的兩類箱靴。
對于一個名為outer的外部類和其內(nèi)部定義的名為inner的內(nèi)部類。編譯完成后出現(xiàn)outer.class和outer$inner.class兩類荷愕。
(編寫一個程序檢驗:在一個TestOuter.java程序中驗證內(nèi)部類在編譯完成之后衡怀,會出現(xiàn)幾個class.)
成員內(nèi)部類不可以有靜態(tài)屬性。(為什么安疗?)
如果在外部類的外部訪問內(nèi)部類抛杨,使用out.inner.
建立內(nèi)部類對象時應注意:
在外部類的內(nèi)部可以直接使用inner s=new inner();(因為外部類知道inner是哪個類,所以可以生成對象荐类。)
而在外部類的外部怖现,要生成(new)一個內(nèi)部類對象,需要首先建立一個外部類對象(外部類可用)玉罐,然后在生成一個內(nèi)部類對象屈嗤。
Outer.Inner in=Outer.new.Inner()。
錯誤的定義方式:
Outer.Inner in=new Outer.Inner()吊输。
注意:當Outer是一個private類時饶号,外部類對于其外部訪問是私有的,所以就無法建立外部類對象季蚂,進而也無法建立內(nèi)部類對象茫船。
局部內(nèi)部類:在方法中定義的內(nèi)部類稱為局部內(nèi)部類琅束。
與局部變量類似,在局部內(nèi)部類前不加修飾符public和private算谈,其范圍為定義它的代碼塊涩禀。
注意:局部內(nèi)部類不僅可以訪問外部類實例變量,還可以訪問外部類的局部變量(但此時要求外部類的局部變量必須為final)然眼?艾船?
在類外不可直接生成局部內(nèi)部類(保證局部內(nèi)部類對外是不可見的)。
要想使用局部內(nèi)部類時需要生成對象罪治,對象調(diào)用方法丽声,在方法中才能調(diào)用其局部內(nèi)部類。
靜態(tài)內(nèi)部類:(注意:前三種內(nèi)部類與變量類似觉义,所以可以對照參考變量)
靜態(tài)內(nèi)部類定義在類中,任何方法外浴井,用static定義晒骇。
靜態(tài)內(nèi)部類只能訪問外部類的靜態(tài)成員。
生成(new)一個靜態(tài)內(nèi)部類不需要外部類成員:這是靜態(tài)內(nèi)部類和成員內(nèi)部類的區(qū)別磺浙。靜態(tài)內(nèi)部類的對象可以直接生成:
Outer.Inner in=new Outer.Inner()洪囤;
而不需要通過生成外部類對象來生成。這樣實際上使靜態(tài)內(nèi)部類成為了一個頂級類撕氧。
靜態(tài)內(nèi)部類不可用private來進行定義瘤缩。例子:
對于兩個類,擁有相同的方法:
People
{
run();
}
Machine{
run();
}
此時有一個robot類:
class Robot extends People implement Machine.
此時run()不可直接實現(xiàn)伦泥。
注意:當類與接口(或者是接口與接口)發(fā)生方法命名沖突的時候剥啤,此時必須使用內(nèi)部類來實現(xiàn)。
用接口不能完全地實現(xiàn)多繼承不脯,用接口配合內(nèi)部類才能實現(xiàn)真正的多繼承府怯。
匿名內(nèi)部類(必須掌握):
匿名內(nèi)部類是一種特殊的局部內(nèi)部類,它是通過匿名類實現(xiàn)接口防楷。
IA被定義為接口牺丙。
IA I=new IA(){};
注:一個匿名內(nèi)部類一定是在new的后面,用其隱含實現(xiàn)一個接口或?qū)崿F(xiàn)一個類复局,沒有類名冲簿,根據(jù)多態(tài),我們使用其父類名亿昏。
因其為局部內(nèi)部類峦剔,那么局部內(nèi)部類的所有限制都對其生效。
匿名內(nèi)部類是唯一一種無構(gòu)造方法類龙优。
匿名內(nèi)部類在編譯的時候由系統(tǒng)自動起名Out$1.class羊异。
如果一個對象編譯時的類型是接口事秀,那么其運行的類型為實現(xiàn)這個接口的類。
因匿名內(nèi)部類無構(gòu)造方法野舶,所以其使用范圍非常的有限易迹。
(下午:)Exception(例外/異常)(教程上的MODEL7)
對于程序可能出現(xiàn)的錯誤應該做出預案。
例外是程序中所有出乎意料的結(jié)果平道。(關系到系統(tǒng)的健壯性)
JAVA會將所有的錯誤封裝成為一個對象睹欲,其根本父類為Throwable。
Throwable有兩個子類:Error和Exception一屋。
一個Error對象表示一個程序錯誤窘疮,指的是底層的、低級的冀墨、不可恢復的嚴重錯誤闸衫。此時程序一定會退出,因為已經(jīng)失去了運行所必須的物理環(huán)境诽嘉。
對于Error錯誤我們無法進行處理蔚出,因為我們是通過程序來應對錯誤,可是程序已經(jīng)退出了虫腋。
我們可以處理的Throwable對象中只有Exception對象(例外/異常)骄酗。
Exception有兩個子類:Runtime exception(未檢查異常)
非Runtime exception(已檢查異常)
(注意:無論是未檢查異常還是已檢查異常在編譯的時候都不會被發(fā)現(xiàn),在編譯的過程中檢查的是程序的語法錯誤悦冀,而異常是一個運行時程序出錯的概念趋翻。)
在Exception中,所有的非未檢查異常都是已檢查異常盒蟆,沒有另外的異常Lだ印!
未檢查異常是因為程序員沒有進行必要的檢查茁影,因為他的疏忽和錯誤而引起的異常宙帝。一定是屬于虛擬機內(nèi)部的異常(比如空指針)。
應對未檢查異常就是養(yǎng)成良好的檢查習慣募闲。
已檢查異常是不可避免的步脓,對于已檢查異常必須實現(xiàn)定義好應對的方法。
已檢查異澈坡荩肯定跨越出了虛擬機的范圍靴患。(比如“未找到文件”)
如何處理已檢查異常(對于所有的已檢查異常都要進行處理):
首先了解異常形成的機制:
當一個方法中有一條語句出現(xiàn)了異常,它就會throw(拋出)一個例外對象要出,然后后面的語句不會執(zhí)行返回上一級方法鸳君,其上一級方法接受到了例外對象之后,有可能對這個異常進行處理患蹂,也可能將這個異常轉(zhuǎn)到它的上一級或颊。
對于接收到的已檢查異常有兩種處理方式:throws和try方法砸紊。
注意:出錯的方法有可能是JDK,也可能是程序員寫的程序囱挑,無論誰寫的醉顽,拋出一定用throw。
例:public void print() throws Exception.
對于方法a平挑,如果它定義了throws Exception游添。那么當它調(diào)用的方法b返回異常對象時,方法a并不處理通熄,而將這個異常對象向上一級返回唆涝,如果所有的方法均不進行處理,返回到主方法唇辨,程序中止廊酣。(要避免所有的方法都返回的使用方法,因為這樣出現(xiàn)一個很小的異常就會令程序中止)赏枚。
如果在方法的程序中有一行throw new Exception()啰扛,返回錯誤,那么其后的程序不執(zhí)行嗡贺。因為錯誤返回后,后面的程序肯定沒有機會執(zhí)行鞍帝,那么JAVA認為以后的程序沒有存在的必要诫睬。
對于try……catch格式:
try{可能出現(xiàn)錯誤的代碼塊}catch(exception e){進行處理的代碼};
對象變量的聲明
用這種方法帕涌,如果代碼正確摄凡,那么程序不經(jīng)過catch語句直接向下運行;
如果代碼不正確蚓曼,則將返回的異常對象和e進行匹配亲澡,如果匹配成功,則處理其后面的異常處理代碼纫版。(如果用exception來聲明e的話床绪,因為exception為所有exception對象的父類,所有肯定匹配成功)其弊。處理完代碼后這個例外就完全處理完畢癞己,程序會接著從出現(xiàn)異常的地方向下執(zhí)行(是從出現(xiàn)異常的地方還是在catch后面呢?利用程序進行驗證)梭伐。最后程序正常退出痹雅。
Try中如果發(fā)現(xiàn)錯誤,即跳出try去匹配catch糊识,那么try后面的語句就不會被執(zhí)行绩社。
一個try可以跟進多個catch語句摔蓝,用于處理不同情況。當一個try只能匹配一個catch。
我們可以寫多個catch語句谍失,但是不能將父類型的exception的位置寫在子類型的excepiton之前饵逐,因為這樣父類型肯定先于子類型被匹配,所有子類型就成為廢話绘盟。JAVA編譯出錯。
在try悯仙,catch后還可以再跟一子句finally龄毡。其中的代碼語句無論如何都會被執(zhí)行(因為finally子句的這個特性,所以一般將釋放資源锡垄,關閉連接的語句寫在里面)沦零。
如果在程序中書寫了檢查(拋出)exception但是沒有對這個可能出現(xiàn)的檢查結(jié)果進行處理,那么程序就會報錯货岭。
而如果只有處理情況(try)而沒有相應的catch子句路操,則編譯還是通不過。
如何知道在編寫的程序中會出現(xiàn)例外呢
調(diào)用方法千贯,查看API中查看方法中是否有已檢查錯誤屯仗。
在編譯的過程中看提示信息,然后加上相應的處理搔谴。
Exception有一個message屬性魁袜。在使用catch的時候可以調(diào)用:
Catch(IOException e){System.out.println(e.message())};
Catch(IOException e){e.printStackTrace()};
上面這條語句回告訴我們出錯類型所歷經(jīng)的過程,在調(diào)試的中非常有用敦第。
開發(fā)中的兩個道理:
①如何控制try的范圍:根據(jù)操作的連動性和相關性峰弹,如果前面的程序代碼塊拋出的錯誤影響了后面程序代碼的運行,那么這個我們就說這兩個程序代碼存在關聯(lián)芜果,應該放在同一個try中鞠呈。
對已經(jīng)查出來的例外,有throw(積極)和try catch(消極)兩種處理方法右钾。
對于try catch放在能夠很好地處理例外的位置(即放在具備對例外進行處理的能力的位置)蚁吝。如果沒有處理能力就繼續(xù)上拋。
當我們自己定義一個例外類的時候必須使其繼承excepiton或者RuntimeException霹粥。
Throw是一個語句灭将,用來做拋出例外的功能。
而throws是表示如果下級方法中如果有例外拋出后控,那么本方法不做處理庙曙,繼續(xù)向上拋出。
Throws后跟的是例外類型浩淘。
斷言是一種調(diào)試工具(assert)
其后跟的是布爾類型的表達式捌朴,如果表達式結(jié)果為真不影響程序運行吴攒。如果為假系統(tǒng)出現(xiàn)低級錯誤,在屏幕上出現(xiàn)assert信息砂蔽。
Assert只是用于調(diào)試洼怔。在產(chǎn)品編譯完成后上線assert代碼就被刪除了。
方法的覆蓋中左驾,如果子類的方法拋出的例外是父類方法拋出的例外的父類型镣隶,那么編譯就會出錯:子類無法覆蓋父類。
結(jié)論:子類方法不可比父類方法拋出更多的例外诡右。子類拋出的例外或者與父類拋出的例外一致安岂,或者是父類拋出例外的子類型》牵或者子類型不拋出例外域那。
如果父類型無throws時,子類型也不允許出現(xiàn)throws猜煮。此時只能使用try catch次员。
練習:寫一個方法:int add(int a,int b)
{
return a+b;
}
當a+b=100;拋出100為異常處理王带。
12.02
集合(從本部分開始涉及API)
集合是指一個對象容納了多個對象淑蔚,這個集合對象主要用來管理維護一系列相似的對象。
數(shù)組就是一種對象愕撰。(練習:如何編寫一個數(shù)組程序束倍,并進行遍歷。)
java.util.*定義了一系列的接口和類盟戏,告訴我們用什么類NEW出一個對象,可以進行超越數(shù)組的操作甥桂。
(注:JAVA1.5對JAVA1.4的最大改進就是增加了對范型的支持)
集合框架接口的分類:(分collection接口和map接口)
Collection接口Map接口
List接口Set接口SortedMap接口
SortedSet接口
JAVA中所有與集合有關的實現(xiàn)類都是這六個接口的實現(xiàn)類柿究。
Collection接口:集合中每一個元素為一個對象,這個接口將這些對象組織在一起黄选,形成一維結(jié)構(gòu)蝇摸。
List接口代表按照元素一定的相關順序來組織(在這個序列中順序是主要的),List接口中數(shù)據(jù)可重復办陷。
Set接口是數(shù)學中集合的概念:其元素無序貌夕,且不可重復。(正好與List對應)
SortedSet會按照數(shù)字將元素排列民镜,為“可排序集合”啡专。
Map接口中每一個元素不是一個對象,而是一個鍵對象和值對象組成的鍵值對(Key-Value)制圈。
Key-Value是用一個不可重復的key集合對應可重復的value集合们童。(典型的例子是字典:通過頁碼的key值找字的value值)畔况。
例子:
key1—value1;
key2—value2;
key3—value3.
SortedMap:如果一個Map可以根據(jù)key值排序,則稱其為SortedMap慧库。(如字典)
!!注意數(shù)組和集合的區(qū)別:數(shù)組中只能存簡單數(shù)據(jù)類型跷跪。Collection接口和Map接口只能存對象。
以下介紹接口:
List接口:(介紹其下的兩個實現(xiàn)類:ArrayList和LinkedList)
ArrayList和數(shù)組非常類似齐板,其底層①也用數(shù)組組織數(shù)據(jù)吵瞻,ArrayList是動態(tài)可變數(shù)組。
底層:指存儲格式甘磨。說明ArrayList對象都是存在于數(shù)組中橡羞。
注:數(shù)組和集合都是從下標0開始。
ArrayList有一個add(Object o)方法用于插入數(shù)組宽档。
ArrayList的使用:(完成這個程序)
先importjava.util.*尉姨;
用ArrayList在一個數(shù)組中添加數(shù)據(jù),并遍歷吗冤。
ArrayList中數(shù)組的順序與添加順序一致又厉。
只有List可用get和size。而Set則不可用(因其無序)椎瘟。
Collection接口都是通過Iterator()(即迭代器)來對Set和List遍歷覆致。
通過語句:Iterator it=c.iterator();得到一個迭代器,將集合中所有元素順序排列肺蔚。然后可以通過interator方法進行遍歷煌妈,迭代器有一個游標(指針)指向首位置。
Interator有hasNext()宣羊,用于判斷元素右邊是否還有數(shù)據(jù)璧诵,返回True說明有。然后就可以調(diào)用next動作仇冯。Next()會將游標移到下一個元素之宿,并把它所跨過的元素返回。(這樣就可以對元素進行遍歷)
練習:寫一個程序苛坚,輸入對象信息比被,比較基本信息。
集合中每一個元素都有對象泼舱,如有字符串要經(jīng)過強制類型轉(zhuǎn)換等缀。
Collections是工具類,所有方法均為有用方法娇昙,且方法為static尺迂。
有Sort方法用于給List排序。
Collections.Sort()分為兩部分,一部分為排序規(guī)則枪狂;一部分為排序算法危喉。
規(guī)則用來判斷對象;算法是考慮如何排序州疾。
對于自定義對象辜限,Sort不知道規(guī)則,所以無法比較严蓖。這種情況下一定要定義排序規(guī)則薄嫡。方式有兩種:
java.lang下面有一個接口:Comparable(可比較的)
可以讓自定義對象實現(xiàn)一個接口,這個接口只有一個方法comparableTo(Object o)
其規(guī)則是當前對象與o對象進行比較颗胡,其返回一個int值毫深,系統(tǒng)根據(jù)此值來進行排序。
如當前對象>o對象毒姨,則返回值>0哑蔫;(可將返回值定義為1)
如當前對象=o對象,則返回值=0弧呐;
如當前對象對象闸迷,則返回值〈0。(可將返回值定義為-1)
看TestArraylist的java代碼俘枫。
我們通過返回值1和-1位置的調(diào)換來實現(xiàn)升序和降序排列的轉(zhuǎn)換腥沽。
java.util下有一個Comparator(比較器)
它擁有compare(),用來比較兩個方法鸠蚪。
要生成比較器今阳,則用Sort中Sort(List,List(Compate))
第二種方法更靈活,且在運行的時候不用編譯茅信。
注意:要想實現(xiàn)comparTo()就必須在主方法中寫上implement comparable.
練習:生成一個EMPLOYEE類盾舌,然后將一系列對象放入到ArrayList。用Iterator遍歷蘸鲸,排序之后矿筝,再進行遍歷。
集合的最大缺點是無法進行類型判定(這個缺點在JAVA1.5中已經(jīng)解決)棚贾,這樣就可能出現(xiàn)因為類型不同而出現(xiàn)類型錯誤。
解決的方法是添加類型的判斷榆综。
LinkedList接口(在代碼的使用過程中和ArrayList沒有什么區(qū)別)
ArrayList底層是object數(shù)組妙痹,所以ArrayList具有數(shù)組的查詢速度快的優(yōu)點以及增刪速度慢的缺點。
而在LinkedList的底層是一種雙向循環(huán)鏈表鼻疮。在此鏈表上每一個數(shù)據(jù)節(jié)點都由三部分組成:前指針(指向前面的節(jié)點的位置)怯伊,數(shù)據(jù),后指針(指向后面的節(jié)點的位置)判沟。最后一個節(jié)點的后指針指向第一個節(jié)點的前指針耿芹,形成一個循環(huán)崭篡。
雙向循環(huán)鏈表的查詢效率低但是增刪效率高。所以LinkedList具有查詢效率低但增刪效率高的特點吧秕。
ArrayList和LinkedList在用法上沒有區(qū)別琉闪,但是在功能上還是有區(qū)別的。
LinkedList經(jīng)常用在增刪操作較多而查詢操作很少的情況下:隊列和堆棧砸彬。
隊列:先進先出的數(shù)據(jù)結(jié)構(gòu)颠毙。
堆棧:后進先出的數(shù)據(jù)結(jié)構(gòu)。
注意:使用堆棧的時候一定不能提供方法讓不是最后一個元素的元素獲得出棧的機會砂碉。
LinkedList提供以下方法:(ArrayList無此類方法)
addFirst();
removeFirst();
addLast();
removeLast();
在堆棧中蛀蜜,push為入棧操作,pop為出棧操作增蹭。
Push用addFirst()滴某;pop用removeFirst(),實現(xiàn)后進先出滋迈。
用isEmpty()--其父類的方法霎奢,來判斷棧是否為空。
在隊列中杀怠,put為入隊列操作椰憋,get為出隊列操作。
Put用addFirst()赔退,get用removeLast()實現(xiàn)隊列橙依。
List接口的實現(xiàn)類(Vector)(與ArrayList相似,區(qū)別是Vector是重量級的組件硕旗,使用使消耗的資源比較多窗骑。)
結(jié)論:在考慮并發(fā)的情況下用Vector(保證線程的安全)。
在不考慮并發(fā)的情況下用ArrayList(不能保證線程的安全)漆枚。
面試經(jīng)驗(知識點):
java.util.stack(stack即為堆棧)的父類為Vector创译。可是stack的父類是最不應該為Vector的墙基。因為Vector的底層是數(shù)組软族,且Vector有get方法(意味著它可能訪問到并不屬于最后一個位置元素的其他元素,很不安全)残制。
對于堆棧和隊列只能用push類和get類立砸。
Stack類以后不要輕易使用。
3醪琛?抛!!實現(xiàn)堆棧一定要用LinkedList。
(在JAVA1.5中螺戳,collection有queue來實現(xiàn)隊列搁宾。)
Set-HashSet實現(xiàn)類:
遍歷一個Set的方法只有一個:迭代器(interator)。
HashSet中元素是無序的(這個無序指的是數(shù)據(jù)的添加順序和后來的排列順序不同)倔幼,而且元素不可重復盖腿。
在Object中除了有final(),toString()凤藏,equals()奸忽,還有hashCode()。
HashSet底層用的也是數(shù)組揖庄。
當向數(shù)組中利用add(Object o)添加對象的時候栗菜,系統(tǒng)先找對象的hashCode:
int hc=o.hashCode();返回的hashCode為整數(shù)值。
Int I=hc%n;(n為數(shù)組的長度)蹄梢,取得余數(shù)后疙筹,利用余數(shù)向數(shù)組中相應的位置添加數(shù)據(jù),以n為6為例禁炒,如果I=0則放在數(shù)組a[0]位置而咆,如果I=1,則放在數(shù)組a[1]位置。如果equals()返回的值為true幕袱,則說明數(shù)據(jù)重復暴备。如果equals()返回的值為false,則再找其他的位置進行比較们豌。這樣的機制就導致兩個相同的對象有可能重復地添加到數(shù)組中涯捻,因為他們的hashCode不同。
如果我們能夠使兩個相同的對象具有相同hashcode望迎,才能在equals()返回為真障癌。
在實例中督赤,定義student對象時覆蓋它的hashcode稽犁。
因為String類是自動覆蓋的桨踪,所以當比較String類的對象的時候如迟,就不會出現(xiàn)有兩個相同的string對象的情況。
現(xiàn)在摸吠,在大部分的JDK中纬凤,都已經(jīng)要求覆蓋了hashCode姿搜。
結(jié)論:如將自定義類用hashSet來添加對象胸墙,一定要覆蓋hashcode()和equals()我注,覆蓋的原則是保證當兩個對象hashcode返回相同的整數(shù),而且equals()返回值為True劳秋。
如果偷懶,沒有設定equals(),就會造成返回hashCode雖然結(jié)果相同玻淑,但在程序執(zhí)行的過程中會多次地調(diào)用equals()嗽冒,從而影響程序執(zhí)行的效率。
我們要保證相同對象的返回的hashCode一定相同补履,也要保證不相同的對象的hashCode盡可能不同(因為數(shù)組的邊界性添坊,hashCode還是可能相同的)。例子:
public int hashCode(){
return name.hashcode()+age;
}
這個例子保證了相同姓名和年齡的記錄返回的hashCode是相同的箫锤。
使用hashSet的優(yōu)點:
hashSet的底層是數(shù)組贬蛙,其查詢效率非常高。而且在增加和刪除的時候由于運用的hashCode的比較開確定添加元素的位置谚攒,所以不存在元素的偏移阳准,所以效率也非常高。因為hashSet查詢和刪除和增加元素的效率都非常高馏臭。
但是hashSet增刪的高效率是通過花費大量的空間換來的:因為空間越大野蝇,取余數(shù)相同的情況就越小。HashSet這種算法會建立許多無用的空間括儒。
使用hashSet接口時要注意绕沈,如果發(fā)生沖突,就會出現(xiàn)遍歷整個數(shù)組的情況帮寻,這樣就使得效率非常的低乍狐。
練習:new一個hashset,插入employee對象固逗,不允許重復浅蚪,并且遍歷出來。
添加知識點:
集合對象存放的是一系列對象的引用抒蚜。
例:
Student S
Al.add(s);
s.setName(“l(fā)ucy”);
Student s2=(Student)(al.get(o1));
可知s2也是s掘鄙。
12.05
SortedSet可自動為元素排序。
SortedSet的實現(xiàn)類是TreeSet:它的作用是字為添加到TreeSet中的元素排序嗡髓。
練習:自定義類用TreeSet排序操漠。
與HashSet不同,TreeSet并不需要實現(xiàn)HashCode()和equals()饿这。
只要實現(xiàn)compareable和compareTo()接可以實現(xiàn)過濾功能浊伙。
(注:HashSet不調(diào)用CompareTo())。
如果要查詢集合中的數(shù)據(jù)长捧,使用Set必須全部遍歷嚣鄙,所以查詢的效率低。使用Map串结,可通過查找key得到value哑子,查詢效率高舅列。
集合中常用的是:ArrayList,HashSet卧蜓,HashMap帐要。其中ArrayList和HashMap使用最為廣泛。
使用HashMap弥奸,put()表示放置元素榨惠,get()表示取元素。
遍歷Map盛霎,使用keySet()可以返回set值赠橙,用keySet()得到key值,使用迭代器遍歷愤炸,然后使用put()得到value值期揪。
上面這個算法的關鍵語句:
Set s=m.keySet();
Interator it=new interator();
Object key=it.next();
Object value=m.get(key);
注意:HashMap與HashCode有關,用Sort對象排序摇幻。
如果在HashMap中有key值重復横侦,那么后面一條記錄的value覆蓋前面一條記錄。
Key值既然可以作為對象绰姻,那么也可以用一個自定義的類枉侧。比如:
m.put(new sutdent(“Liucy”,30),”boss”)
如果沒有語句來判定Student類對象是否相同,則會全部打印出來狂芋。
當我們用自定義的類對象作為key時榨馁,我們必須在程序中覆蓋HashCode()和equals()。
注:HashMap底層也是用數(shù)組帜矾,HashSet底層實際上也是HashMap翼虫,HashSet類中有HashMap屬性(我們?nèi)绾卧?/b>API中查屬性)。HashSet實際上為(key.null)類型的HashMap屡萤。有key值而沒有value值珍剑。
正因為以上的原因,TreeSet和TreeMap的實現(xiàn)也有些類似的關系死陆。
注意:TreeSet和TreeMap非常的消耗時間招拙,因此很少使用。
我們應該熟悉各種實現(xiàn)類的選擇——非常體現(xiàn)你的功底措译。
HashSet VS TreeSet:HashSet非常的消耗空間别凤,TreeSet因為有排序功能,因此資源消耗非常的高领虹,我們應該盡量少使用规哪,而且最好不要重復使用。
基于以上原因塌衰,我們盡可能的運用HashSet而不用TreeSet诉稍,除非必須排序蝠嘉。
同理:HashMapVSTreeMap:一般使用HashMap,排序的時候使用TreeMap杯巨。
HashMap VS Hashtable(注意在這里table的第一個字母小寫)之間的區(qū)別有些類似于ArrayList和Vector是晨,Hashtable是重量級的組件,在考慮并發(fā)的情況舔箭,對安全性要求比較高的時候使用。
Map的運用非常的多蚊逢。
使用HashMap()层扶,如果使用自定義類,一定要覆蓋HashCode()和equals()烙荷。
重點掌握集合的四種操作:增加镜会、刪除、遍歷终抽、排序戳表。
Module8—12利用兩天的時間完成。
Module8:圖形界面
Module9:事件模型(在本部分最重要)
Module10:AWT
Module11:Swing
Module12:Applet(這個技術基本已經(jīng)被淘汰)
軟件應用的三個發(fā)展階段:
單機應用
網(wǎng)絡應用(C/S結(jié)構(gòu))
BS結(jié)構(gòu):B表示瀏覽器昼伴,S表示server端匾旭。即利用瀏覽器作為客戶端,因此對于圖形界面的要求已經(jīng)不高圃郊,現(xiàn)在的發(fā)展趨勢是不使用安裝价涝,即不用任何的本地應用,圖形很快就會被服務器構(gòu)件開發(fā)所取代持舆。
經(jīng)驗之談:Swing的開發(fā)工作會非常的累色瘩,而且這項技術正在走向沒落。避免從事有這種特征的工作逸寓。
AWT也即將被取代居兆。
Module8—Module11所使用的技術都將被JSF技術所取代。
JSF是服務器端的Swing:目前技術已經(jīng)成熟竹伸,但是開發(fā)環(huán)境(工具)還不成熟泥栖。
Module12的Applet技術也將被WebStart所取代。
Module9為重點佩伤,所謂事件模型是指觀察者設計模式的JAVA應用聊倔。事件模型是重點。
Module8:圖形界面(java.awt.*)
Awt:抽象窗口工具箱生巡,它由三部分組成:
①組件:界面元素耙蔑;
②容器:裝載組件的容器(例如窗體);
③布局管理器:負責決定容器中組件的擺放位置孤荣。
圖形界面的應用分四步:
選擇一個容器:
⑴window:帶標題的容器(如Frame)甸陌;
⑵Panel:面板
通過add()想容器中添加組件须揣。
Java的圖形界面依然是跨平臺的。但是在調(diào)用了一個窗體之后只生成一個窗體钱豁,沒有事件的處理耻卡,關閉按鈕并不工作。此時只能使用CTRL+C終止程序牲尺。
②設置一個布局管理器:用setLayout()卵酪;
③向容器中添加組件;
添加組件的事務處理谤碳。P198
P204:Panel也是一種容器:但是不可見的溃卡。在設置容易的時候不要忘記設置它們的可見性。
Panel pan=new Panel;
Fp.setLayout(null);表示不要布局管理器蜒简。
五種布局管理器:
P206:Flow Layout(流式布局):按照組件添加到容器中的順序瘸羡,順序排放組件位置。默認為水平排列搓茬,如果越界那么會向下排列犹赖。排列的位置隨著容器大小的改變而改變。
Panel默認的布局管理器為Flow Layout卷仑。
Border Layout:會將容器非常五個區(qū)域:東西南北中峻村。
語句:
Button b1=new Botton(“north”);//botton上的文字
f.add(b1,”North”);//表示b1這個botton放在north位置
注:一個區(qū)域只能放置一個組件,如果想在一個區(qū)域放置多個組件就需要使用Panel來裝載锡凝。
Frame和Dialog的默認布局管理器是Border Layout雀哨。
Grid Layout:將容器生成等長等大的條列格,每個塊中放置一個組件私爷。
f.setLayout GridLayout(5,2,10,10)//表示條列格為5行2類雾棺,后面為格間距。
CardLayout:一個容器可以放置多個組件衬浑,但每次只有一個組件可見(組件重疊)捌浩。
使用first(),last()工秩,next()可以決定哪個組件可見尸饺。可以用于將一系列的面板有順序地呈現(xiàn)給用戶助币。
重點:GridBag Layout:在Grid中可指定一個組件占據(jù)多行多列浪听,GridBag的設置非常的煩瑣。
Module9:AWT:事件模型
事件模型指的是對象之間進行通信的設計模式眉菱。
對象1給對象2發(fā)送一個信息相當于對象1引用對象2的方法迹栓。
模型即是一種設計模式(約定俗成)
對象對為三種:
①事件源:發(fā)出事件者;
②事件對象:發(fā)出的事件本身俭缓;
事件監(jiān)聽器:提供處理事件指定的方法克伊。
Java AWT事件模型也稱為授權事件模型酥郭,指事件可以和監(jiān)聽器之間事先建立一種關系:約定那些事件如何處理,由誰去進行處理愿吹。這種約定稱為授權不从。
一個事件源可以授權多個監(jiān)聽者(授權也稱為監(jiān)聽者的注冊);
多個事件源也可以注冊多個事件監(jiān)聽器犁跪。
監(jiān)聽者對于事件源的發(fā)出的事件作出響應椿息。
在java.util中有EventListener接口:所有事件監(jiān)聽者都要實現(xiàn)這個接口。
java.util中有EventObject類:所有的事件都為其子類坷衍。
事件范例在\CoreJava\Girl.java文件中撵颊。(文件已加注釋)
注意:接口因?qū)Σ煌氖录O(jiān)聽器對其處理可能不同,所以只能建立監(jiān)聽的功能惫叛,而無法實現(xiàn)處理。
下面程序建立監(jiān)聽功能:
//監(jiān)聽器接口要定義監(jiān)聽器所具備的功能逞刷,定義方法
{
void WhatIdoWhenGirlHappy(EmotionEvent e);
void WhatIdoWhenGirlSad(EmotionEvent e);
}
注意查看參考書:事件的設置模式嘉涌,如何實現(xiàn)授權模型。
事件模式的實現(xiàn)步驟:
開發(fā)事件對象(事件發(fā)送者)——接口——接口實現(xiàn)類——設置監(jiān)聽對象
一定要理解透徹Gril.java程序夸浅。
重點:學會處理對一個事件源有多個事件的監(jiān)聽器(在發(fā)送消息時監(jiān)聽器收到消息的排名不分先后)仑最。
事件監(jiān)聽的響應順序是不分先后的,不是誰先注冊誰就先響應帆喇。
事件監(jiān)聽由兩個部分組成(接口和接口的實現(xiàn)類)警医。
事件源事件對象事件監(jiān)聽
grilEmotinEventEmotionListener(接口)、Boy(接口的實現(xiàn)類)
鼠標事件:MouseEvent坯钦,接口:MouseListener预皇。
P235 ActionEvent。
注意在寫程序的時候:import java.awt.*;以及import java.awt.event.*注意兩者的不同婉刀。
在生成一個窗體的時候吟温,點擊窗體的右上角關閉按鈕激發(fā)窗體事件的方法:窗體Frame為事件源,WindowsListener接口調(diào)用Windowsclosing()突颊。
為了配合后面的實現(xiàn)鲁豪,我們必須將WindowsListener所有的方法都實現(xiàn),除了Windowsclosing方法律秃,其余的方法均為空實現(xiàn)爬橡。
(練習:寫一個帶button窗體,點關閉按鈕退出棒动。)
上面程序中實現(xiàn)了許多不必要的實現(xiàn)類糙申,雖然是空實現(xiàn)。
為了避免上面那些無用的實現(xiàn)船惨,可以利用WindowEvent的一個WindowEvent類郭宝,還是利用windowsListener辞槐。還有WindowAdapter類,它已經(jīng)實現(xiàn)了WindowsListener粘室。它給出的全部都是空實現(xiàn)榄檬,那就可以只寫想要實現(xiàn)的類,去覆蓋其中的類衔统,就不用寫空實現(xiàn)鹿榜。
注意:監(jiān)聽過多,會拋tooManyListener例外锦爵。
12.06
Module10
Canvas組件:畫布舱殿,可以實現(xiàn)動畫操作。
TextArea:文本域险掀。
在單行文本域中回車會激發(fā)ActionEvent沪袭。
用CheckBoxGroup實現(xiàn)單選框功能。
Java中樟氢,單選框和復選框都是使用CheckBox實現(xiàn)冈绊。
菜單:new MenuBar(),MenuBar表示菜單條埠啃。
菜單中的每一項為MenuItem死宣,一般級聯(lián)菜單不應該超過三級涌哲。
練習:
設計一個計算器:注意設置一個boolean值(append)來判斷輸入數(shù)字是位于第一個數(shù)的后面還是屬于輸入的第二個數(shù)儒溉。
設置一個變量來存放“+”,點完運算符后边灭,將append設置為false潦牛。
String number1
Charoperator存放運算符眶掌。
Module11 Swing
AWT是Java最早出現(xiàn)的圖形界面,但很快就被Swing所取代巴碗。
Swing才是一種真正的圖形開發(fā)畏线。
AWT在不同平臺所出現(xiàn)的界面可能有所不同:因為每個OS都有自己的UI組件庫,java調(diào)用不同系統(tǒng)的UI良价。
注意AWT為重量級組件寝殴,相當消耗資源,且不同系統(tǒng)的組件可能不同明垢。因為這個問題使得AWT開發(fā)的軟件難以作到跨平臺蚣常。
更為要命的是:不同OS的組件庫都存在BUG。必須多種平臺進行測試痊银,并且AWT的組件庫并不豐富抵蚊。
為解決以上問題,SUN和IBM以及NETSCAPE聯(lián)合開發(fā)出JAVA基礎類包Swing:注意JAVA的基礎類以Swing為核心。
注意引用:javax.swing.*;javax表示JAVA的擴展贞绳。
我們在學習JDBC的時候會過度到J2EE谷醉。
在Swing的組件中,基本上都是在AWT組件的名稱前面加“J”冈闭。
一般情況下俱尼,除了Choise等組件:
import javax.swing.*;好要加上:import java.awt.*以及import java.awt.event.*。
Swing與AWT的最大區(qū)別是Swing為JAVA自身的組件萎攒。已經(jīng)不是對等實體遇八,與底層的OS無關。
(JBUILDER就是使用Swing寫的)
Swing與AWT在事件模型處理上是一致的耍休。
Jframe實際上是一堆窗體的疊加刃永。
Swing比AWT更加復雜且靈活。
在JDK1.4中羊精,給JFRAME添加Button不可用jf.add(b)斯够。而是使用jf.getContentPane().add(b)。
content是先申請面板喧锦。不過在JDK1.5中可以使用add.读规。
Jpanel支持雙緩沖技術。
在Jbutton中可以添加圖標裸违。
JscrollPane可以管理比屏幕還要大的組件。
TextArea只有裝入JscrollPane中才能實現(xiàn)滾動條本昏。
JeditorPane用于顯示瀏覽器供汛。
注意:Tabbed Panel與Border的比較。
進度條:ProgressBar涌穆。
JcomboBox:下拉菜單:在AWT中同類組件是choice怔昨。
JlistPanel:選擇列表
BorderPanel:設置邊框
JsplitPanel:可將容器分為兩個部分,其中一個部分有Jtree宿稀。
TextBox:也是一種新的容器趁舀,可以設置組件的間距。
TextFileChoose:文件選擇器祝沸。
ColorChoose:顏色選擇器
Module 12Applet
Applet為Panel的子類
Applet是java的自動執(zhí)行方式(這是它的優(yōu)勢矮烹,主要用于HTML)。
工作四種語法:init()罩锐,start()奉狈,stop(),destory()涩惑。
Swing中有一個Japplet仁期,如使用Swing組件。
Applet消亡的原因:
①java為安全起見對Applet有所限制:Applet不允許訪問本地文件信息、敏感信息跛蛋,不能執(zhí)行本地指令(比如FORMAT)熬的,不能訪問初原服務器之外的其他服務器。
IE不支持新版本的Applet赊级。
Applet的優(yōu)勢:
網(wǎng)絡傳輸押框,自動下載。
Application的優(yōu)勢:沒有執(zhí)行限制此衅。
WebStart:可在網(wǎng)絡傳輸强戴,并且在本地無限制。因此前景光明挡鞍。
練習:
使用Swing實現(xiàn)一個界面骑歹,分為上下兩個部分,南邊為JtextField組件墨微,可編輯道媚,上面為JtextArea組件,不可編輯翘县,在JtextField組件輸入字符最域,按回車,就可以將內(nèi)容輸入到JtextArea組件锈麸。(AREA區(qū)域可以滾動)
12.07
多線程
進程:任務
任務并發(fā)執(zhí)行是一個宏觀概念镀脂,微觀上是串行的。
進程的調(diào)度是有OS負責的(有的系統(tǒng)為獨占式忘伞,有的系統(tǒng)為共享式薄翅,根據(jù)重要性,進程有優(yōu)先級)氓奈。
由OS將時間分為若干個時間片翘魄。
JAVA在語言級支持多線程。
分配時間的仍然是OS舀奶。
參看P377
線程由兩種實現(xiàn)方式:
第一種方式:
class MyThread extends Thread{
public void run(){
需要進行執(zhí)行的代碼暑竟,如循環(huán)。
}
}
public class TestThread{
main(){
Thread t1=new Mythread();
T1.start();
}
}
只有等到所有的線程全部結(jié)束之后育勺,進程才退出但荤。
第二種方式:
Class MyThreadimplements Runnable{
Publicvoid run(){
Runnable target=new MyThread();
Thread t3=new Thread(target);
Thread.start();//啟動線程
}
}
P384:通過接口實現(xiàn)繼承
練習:寫兩個線程:
輸入200個“###”②輸入200個“***”
下面為線程中的7中非常重要的狀態(tài):(有的書上也只有認為前五種狀態(tài):而將“鎖池”和“等待隊列”都看成是“阻塞”狀態(tài)的特殊情況:這種認識也是正確的,但是將“鎖池”和“等待隊列”單獨分離出來有利于對程序的理解)
①⑴
②⑵
③⑶run()結(jié)束
Start()
OS分配CPU
CPU時間片結(jié)束
yield()o.wait()
等待鎖標記
notify()
注意:圖中標記依次為
①輸入完畢涧至;②wake up③t1退出
⑴如等待輸入(輸入設備進行處理纱兑,而CUP不處理),則放入阻塞化借,直到輸入完畢潜慎。
⑵線程休眠sleep()
⑶t1.join()指停止main(),然后在某段時間內(nèi)將t1加入運行隊列,直到t1退出铐炫,main()才結(jié)束垒手。
特別注意:①②③與⑴⑵⑶是一一對應的。
進程的休眠:Thread sleep(1000);//括號中以毫秒為單位
當main()運行完畢倒信,即使在結(jié)束時時間片還沒有用完科贬,CPU也放棄此時間片,繼續(xù)運行其他程序鳖悠。
Try{Thread.sleep(1000);}
Catch(Exception e){e.printStackTrace(e);}
T1.join()表示運行線程放棄執(zhí)行權榜掌,進入阻塞狀態(tài)。
當t1結(jié)束時乘综,main()可以重新進入運行狀態(tài)憎账。
T1.join實際上是把并發(fā)的線程編程并行運行。
線程的優(yōu)先級:1-10卡辰,越大優(yōu)先級越高胞皱,優(yōu)先級越高被OS選中的可能性就越大。(不建議使用九妈,因為不同操作系統(tǒng)的優(yōu)先級并不相同反砌,使得程序不具備跨平臺性,這種優(yōu)先級只是粗略地劃分)萌朱。
注:程序的跨平臺性:除了能夠運行宴树,還必須保證運行的結(jié)果。
一個使用yield()就馬上交出執(zhí)行權晶疼,回到可運行狀態(tài)酒贬,等待OS的再次調(diào)用。
下午:
程序員需要關注的線程同步和互斥的問題冒晰。
多線程的并發(fā)一般不是程序員決定同衣,而是由容器決定竟块。
多線程出現(xiàn)故障的原因:
兩個線程同時訪問一個數(shù)據(jù)資源(臨界資源)壶运,形成數(shù)據(jù)發(fā)生不一致和不完整。
數(shù)據(jù)的不一致往往是因為一個線程中的兩個關聯(lián)的操作只完成了一步浪秘。
避免以上的問題可采用對數(shù)據(jù)進行加鎖的方法
每個對象除了屬性和方法蒋情,都有一個monitor(互斥鎖標記),用來將這個對象交給一個線程耸携,只有拿到monitor的線程才能夠訪問這個對象棵癣。
Synchronized:這個修飾詞可以用來修飾方法和代碼塊
Object obj;
Obj.setValue(123);
Synchronized用來修飾方法,表示當某個線程調(diào)用這個方法之后夺衍,其他的事件不能再調(diào)用這個方法狈谊。只有拿到obj標記的線程才能夠執(zhí)行代碼塊。
注意:Synchronized一定使用在一個方法中。
鎖標記是對象的概念河劝,加鎖是對對象加鎖壁榕,目的是在線程之間進行協(xié)調(diào)。
當用Synchronized修飾某個方法的時候赎瞎,表示該方法都對當前對象加鎖牌里。
給方法加Synchronized和用Synchronized修飾對象的效果是一致的。
一個線程可以拿到多個鎖標記务甥,一個對象最多只能將monitor給一個線程牡辽。
Synchronized是以犧牲程序運行的效率為代價的,因此應該盡量控制互斥代碼塊的范圍敞临。
方法的Synchronized特性本身不會被繼承态辛,只能覆蓋。
線程因為未拿到鎖標記而發(fā)生的阻塞不同于前面五個基本狀態(tài)中的阻塞哟绊,稱為鎖池因妙。
每個對象都有自己的一個鎖池的空間,用于放置等待運行的線程票髓。
這些線程中哪個線程拿到鎖標記由系統(tǒng)決定攀涵。
鎖標記如果過多,就會出現(xiàn)線程等待其他線程釋放鎖標記洽沟,而又都不釋放自己的鎖標記供其他線程運行的狀況以故。就是死鎖。
死鎖的問題通過線程間的通信的方式進行解決裆操。
線程間通信機制實際上也就是協(xié)調(diào)機制怒详。
線程間通信使用的空間稱之為對象的等待隊列,則個隊列也是屬于對象的空間的踪区。
Object類中又一個wait()昆烁,在運行狀態(tài)中,線程調(diào)用wait()缎岗,此時表示著線程將釋放自己所有的鎖標記静尼,同時進入這個對象的等待隊列。
等待隊列的狀態(tài)也是阻塞狀態(tài)传泊,只不過線程釋放自己的鎖標記鼠渺。
Notify()
如果一個線程調(diào)用對象的notify(),就是通知對象等待隊列的一個線程出列眷细。進入鎖池拦盹。如果使用notifyall()則通知等待隊列中所有的線程出列。
注意:只能對加鎖的資源進行wait()和notify()溪椎。
釋放鎖標記只有在Synchronized代碼結(jié)束或者調(diào)用wait()普舆。
注意鎖標記是自己不會自動釋放恬口,必須有通知。
注意在程序中判定一個條件是否成立時要注意使用WHILE要比使用IF要嚴密沼侣。
WHILE會放置程序饒過判斷條件而造成越界楷兽。
補充知識:
suspend()是將一個運行時狀態(tài)進入阻塞狀態(tài)(注意不釋放鎖標記)』伲恢復狀態(tài)的時候用resume()芯杀。Stop()指釋放全部。
這幾個方法上都有Deprecated標志雅潭,說明這個方法不推薦使用揭厚。
一般來說,主方法main()結(jié)束的時候線程結(jié)束扶供,可是也可能出現(xiàn)需要中斷線程的情況筛圆。對于多線程一般每個線程都是一個循環(huán),如果中斷線程我們必須想辦法使其退出椿浓。
如果主方法main()想結(jié)束阻塞中的線程(比如sleep或wait)
那么我們可以從其他進程對線程對象調(diào)用interrupt()太援。用于對阻塞(或鎖池)會拋出例外Interrupted Exception。
這個例外會使線程中斷并執(zhí)行catch中代碼扳碍。
多線程中的重點:實現(xiàn)多線程的兩種方式提岔,Synchronized,以及生產(chǎn)者和消費者問題(ProducerConsumer.java文件)。
練習:
存車位的停開車的次序輸出問題笋敞;
寫兩個線程碱蒙,一個線程打印1-52,另一個線程答應字母A-Z夯巷。打印順序為12A34B56C……5152Z赛惩。通過使用線程之間的通信協(xié)調(diào)關系。
注:分別給兩個對象構(gòu)造一個對象o趁餐,數(shù)字每打印兩個或字母每打印一個就執(zhí)行o.wait()喷兼。在o.wait()之前不要忘了寫o.notify()。
補充說明:通過Synchronized后雷,可知Vector較ArrayList方法的區(qū)別就是Vector所有的方法都有Synchronized季惯。所以Vector更為安全。
同樣:Hashtable較HashMap也是如此喷面。
12.08
Module 10:I/O流(java如何實現(xiàn)與外界數(shù)據(jù)的交流)
Input/Output:指跨越出了JVM的邊界星瘾,與外界數(shù)據(jù)的源頭或者目標數(shù)據(jù)源進行數(shù)據(jù)交換走孽。
輸出
輸入
注意:輸入/輸出是針對JVM而言惧辈。
File類(java.io.*)可表示一個文件,也有可能是一個目錄(在JAVA中文件和目錄都屬于這個類中磕瓷,而且區(qū)分不是非常的明顯)盒齿。
Java.io下的方法是對磁盤上的文件進行磁盤操作念逞,但是無法讀取文件的內(nèi)容。
注意:創(chuàng)建一個文件對象和創(chuàng)建一個文件在JAVA中是兩個不同的概念边翁。前者是在虛擬機中創(chuàng)建了一個文件翎承,但卻并沒有將它真正地創(chuàng)建到OS的文件系統(tǒng)中,隨著虛擬機的關閉符匾,這個創(chuàng)建的對象也就消失了叨咖。而創(chuàng)建一個文件才是在系統(tǒng)中真正地建立一個文件。
例如:File f=new File(“11.txt”);//創(chuàng)建一個名為11.txt的文件對象
f.CreateNewFile();//真正地創(chuàng)建文件
f.CreateMkdir():創(chuàng)建目錄
f.delete()啊胶;刪除文件
f.deleteOnExit();在進程退出的時候刪除文件甸各,這樣的操作通常用在臨時文件的刪除。
對于命令:File f2=new file(“d:\\abc\\789\\1.txt”)
這個命令不具備跨平臺性焰坪,因為不同的OS的文件系統(tǒng)很不相同趣倾。
如果想要跨平臺,在file類下有separtor()某饰,返回鎖出平臺的文件分隔符儒恋。
File.fdir=new File(File.separator);
String str=”abc”+File.separator+”789”;
使用文件下的方法的時候一定注意是否具備跨平臺性。
List():顯示文件的名(相對路徑)
ListFiles():返回Files類型數(shù)組黔漂,可以用getName()來訪問到文件名诫尽。
使用isDirectory()和isFile()來判斷究竟是文件還是目錄。
練習:
寫一個javaTest程序炬守,列出所有目錄下的*.java文件箱锐,把子目錄下的JAVA文件也打印出來。
使用I/O流訪問file中的內(nèi)容劳较。
JVM與外界通過數(shù)據(jù)通道進行數(shù)據(jù)交換驹止。
分類:
按流分為輸入流和輸出流;
按傳輸單位分為字節(jié)流和字符流;
還可以分為節(jié)點流和過濾流盆驹。
節(jié)點流:負責數(shù)據(jù)源和程序之間建立連接教藻;
過濾流:用于給節(jié)點增加功能。
過濾流的構(gòu)造方式是以其他流位參數(shù)構(gòu)造(這樣的設計模式稱為裝飾模式)抖仅。
字節(jié)輸入流:io包中的InputStream為所有字節(jié)輸入流的父類。
Int read();讀入一個字節(jié)(每次一個)砖第;
可先使用newbyte[]=數(shù)組撤卢,調(diào)用read(byte[] b)
read (byte[])返回值可以表示有效數(shù);read (byte[])返回值為-1表示結(jié)束梧兼。
字節(jié)輸出流:io包中的OutputStream位所有字節(jié)輸入流的父類放吩。
Write和輸入流中的read相對應。
在流中close()方法由程序員控制羽杰。因為輸入輸出流已經(jīng)超越了VM的邊界渡紫,所以有時可能無法回收資源到推。
原則:凡是跨出虛擬機邊界的資源都要求程序員自己關閉,不要指望垃圾回收惕澎。
以Stream結(jié)尾的類都是字節(jié)流莉测。
如果構(gòu)造FileOutputStream的同時磁盤會建立一個文件。如果創(chuàng)建的文件與磁盤上已有的文件名重名唧喉,就會發(fā)生覆蓋捣卤。
用FileOutputStream中的boolean,則視八孝,添加情況腌零,將數(shù)據(jù)覆蓋重名文件還是將輸入內(nèi)容放在文件的后面。(編寫程序驗證)
DataOutputStream:輸入數(shù)據(jù)的類型唆阿。
因為每中數(shù)據(jù)類型的不同益涧,所以可能會輸出錯誤。
所有對于:DataOutputStream
DataInputStream
兩者的輸入順序必須一致驯鳖。
過濾流:
bufferedOutputStream
bufferedInputStream
用于給節(jié)點流增加一個緩沖的功能闲询。
在VM的內(nèi)部建立一個緩沖區(qū),數(shù)據(jù)先寫入緩沖區(qū)浅辙,等到緩沖區(qū)的數(shù)據(jù)滿了之后再一次性寫出扭弧,效率很高。
使用帶緩沖區(qū)的輸入輸出流的速度會大幅提高记舆,緩沖區(qū)越大鸽捻,效率越高。(這是典型的犧牲空間換時間)
切記:使用帶緩沖區(qū)的流泽腮,如果數(shù)據(jù)數(shù)據(jù)輸入完畢御蒲,使用flush方法將緩沖區(qū)中的內(nèi)容一次性寫入到外部數(shù)據(jù)源。用close()也可以達到相同的效果诊赊,因為每次close都會使用flush厚满。一定要注意關閉外部的過濾流。
(非重點)管道流:也是一種節(jié)點流碧磅,用于給兩個線程交換數(shù)據(jù)碘箍。
PipedOutputStream
PipedInputStream
輸出流:connect(輸入流)
RondomAccessFile類允許隨機訪問文件
GetFilepoint()可以知道文件中的指針位置,使用seek()定位鲸郊。
Mode(“r”:隨機讀丰榴;”w”:隨機寫;”rw”:隨機讀寫)
練習:寫一個類A秆撮,JAVA A file1 file2
file1要求是系統(tǒng)中已經(jīng)存在的文件四濒。File2是還沒有存在的文件。
執(zhí)行完這個命令,那么file2就是file1中的內(nèi)容峻黍。
字符流:reader\write只能輸純文本文件。
FileReader類:字符文件的輸出
字節(jié)流與字符流的區(qū)別:
字節(jié)流的字符編碼:
字符編碼把字符轉(zhuǎn)換成數(shù)字存儲到計算機中拨匆,按ASCii將字母映射為整數(shù)姆涩。
把數(shù)字從計算機轉(zhuǎn)換成相應的字符的過程稱為解碼。
編碼方式的分類:
ASCII(數(shù)字惭每、英文):1個字符占一個字節(jié)(所有的編碼集都兼容ASCII)
ISO8859-1(歐洲):1個字符占一個字節(jié)
GB-2312/GBK:1個字符占兩個字節(jié)
Unicode: 1個字符占兩個字節(jié)(網(wǎng)絡傳輸速度慢)
UTF-8:變長字節(jié)骨饿,對于英文一個字節(jié),對于漢字兩個或三個字節(jié)台腥。
原則:保證編解碼方式的統(tǒng)一宏赘,才能不至于出現(xiàn)錯誤。
Io包的InputStreamread稱為從字節(jié)流到字符流的橋轉(zhuǎn)換類黎侈。這個類可以設定字符轉(zhuǎn)換方式察署。
OutputStreamred:字符到字節(jié)
Bufferread有readline()使得字符輸入更加方便。
在I/O流中峻汉,所有輸入方法都是阻塞方法贴汪。
Bufferwrite給輸出字符加緩沖,因為它的方法很少休吠,所以使用父類printwrite扳埂,它可以使用字節(jié)流對象,而且方法很多瘤礁。
練習:做一個記事本
swing/JfileChoose: getSelect file()
InputStreeamReader:把字節(jié)變?yōu)樽址?/b>
JAVA中對字符串長無限制bufferedReader(ir)
12.09
class ObjectOutputStream也是過濾流阳懂,使節(jié)點流直接獲得輸出對象。
最有用的方法:WriteObject(Object b)
用流傳輸對象稱為對象的序列化柜思,但并不使所有的對象都可以進行序列化的岩调。只有在實現(xiàn)類時必須實現(xiàn)一個接口:IO包下的Serializable(可序列化的)。此接口沒有任何的方法赡盘,這樣的接口稱為標記接口誊辉。
Class Student implements Serializable
把對象通過流序列化到某一個持久性介質(zhì)稱為對象的可持久化。
Hibernate就是研究對象的可持久化亡脑。
ObuectInputStreamin=newObjectInputStream;
Object o1=in.readObuect();
Student s1=(Student)o1堕澄;
注意:因為o1是一個對象,因為需要對其進行保存霉咨。
Transient用來修飾屬性蛙紫。
Transient int num;
表示當我們對屬性序列化時忽略這個屬性(即忽略不使之持久化)。
所有屬性必須都是可序列化的途戒,特別是當有些屬性本身也是對象的時候坑傅,要尤其注意這一點。
判斷是否一個屬性或?qū)ο罂尚蛄谢?/b>Serialver喷斋。
Serialver TestObject(TestObject必須為已經(jīng)編譯)
執(zhí)行結(jié)果:如果不可序列化唁毒;則出現(xiàn)不可序列化的提示蒜茴。如果可以序列化,那么就會出現(xiàn)序列化的ID:UID浆西。
java.until.*有
StringTokenizer(參數(shù)1粉私,參數(shù)2)按某種符號隔開文件
StringTokenizer(s,”:”)用“:”隔開字符,s為對象近零。
練習:將一個類序列化到文件诺核,然后讀出。下午:
網(wǎng)絡基礎知識
JAVA網(wǎng)絡編程
網(wǎng)絡與分布式集群系統(tǒng)的區(qū)別:每個節(jié)點都是一臺計算機久信,而不是各種計算機內(nèi)部的功能設備窖杀。
Ip:具有全球唯一性,相對于internet裙士,IP為邏輯地址入客。
端口(port):一臺PC中可以有65536個端口,進程通過端口交換數(shù)據(jù)腿椎。連線的時候需要輸入IP也需要輸入端口信息痊项。
計算機通信實際上的主機之間的進程通信,進程的通信就需要在端口進行聯(lián)系酥诽。
192.168.0.23:21
協(xié)議:為了進行網(wǎng)絡中的數(shù)據(jù)交換(通信)而建立的規(guī)則鞍泉、標準或約定。
不同層的協(xié)議是不同的肮帐。
網(wǎng)絡層:尋址咖驮、路由(指如何到達地址的過程)
傳輸層:端口連接
TCP模型:應用層/傳輸層/網(wǎng)絡層/網(wǎng)絡接口
端口是一種抽象的軟件結(jié)構(gòu),與協(xié)議相關:TCP23端口和UDT23端口為兩個不同的概念训枢。
端口應該用1024以上的端口托修,以下的端口都已經(jīng)設定功能。
套接字(socket)的引入:
Ip+Port=Socket(這是個對象的概念恒界。)
Socket為傳輸層概念睦刃,而JSP是對應用層編程。例:
java.net.*;
(Server端定義順序)
ServerSocket(intport)
Socket.accept()十酣;//阻塞方法涩拙,當客戶端發(fā)出請求是就恢復
如果客戶端收到請求:
則Socket SI=ss.accept();
注意客戶端和服務器的Socket為兩個不同的socket耸采。
Socket的兩個方法:
getInputStream():客戶端用
getOutputStream()服務器端用
使用完畢后切記Socket.close()兴泥,兩個Socket都關,而且不用關內(nèi)部的流虾宇。
在client端搓彻,Socket s=new Socket(“127.0.0.1”,8000);
127.0.0.1為一個默認本機的地址。
練習:
客戶端向服務器發(fā)出一個字符串,服務器轉(zhuǎn)換成大寫傳回客戶端旭贬。
大寫的函數(shù):String.toUpperCase()
服務器告訴客戶端:“自開機以來你是第n個用戶”怔接。
12.12
UDP編程:
DatagramSocket(郵遞員):對應數(shù)據(jù)報的Socket概念,不需要創(chuàng)建兩個socket稀轨,不可使用輸入輸出流扼脐。
DatagramPacket(信件):數(shù)據(jù)包,是UDP下進行傳輸數(shù)據(jù)的單位靶端,數(shù)據(jù)存放在字節(jié)數(shù)組中谎势。
UDP也需要現(xiàn)有Server端凛膏,然后再有Client端杨名。
兩端都是DatagramPacket(相當于電話的概念),需要NEW兩個DatagramPacket猖毫。
InetAddress:網(wǎng)址
這種信息傳輸方式相當于傳真台谍,信息打包,在接受端準備紙吁断。
模式:
發(fā)送端:Server:
DatagramPacketinDataPacket=new DatagramPacket ((msg,msg.length); InetAdress.getByName(ip),port);
接收端:
clientAddress=inDataPack.getAddress();//取得地址
clientPort=inDataPack.getPort();//取得端口號
datagramSocket.send;//Server
datagramSocket.accept;//Client
URL:在應用層的編程
注意比較:
http://Localhost:8080/directory//查找網(wǎng)絡服務器的目錄
file://directory//查找本地的文件系統(tǒng)
java的開發(fā)主要以http為基礎趁蕊。
反射:主要用于工具和框架的開發(fā)。
反射是對于類的再抽象仔役;通過字符串來抽象類掷伙。
JAVA類的運行:classLoader:加載到虛擬機(vm)
Vm中只能存儲對象(動態(tài)運行時的概念),.class文件加載到VM上就成為一個對象又兵,同時初始靜態(tài)成員及靜態(tài)代碼(只執(zhí)行一次)任柜。
Lang包下有一個類為Class:在反射中使用。此類中的每個對象為VM中的類對象沛厨,每個類都對應類類的一個對象(class.class)宙地。
例:對于一個Object類,用getClass()得到其類的對象逆皮,獲得類的對象就相當于獲得類的信息宅粥,可以調(diào)用其下的所有方法,包括類的私有方法电谣。
注意:在反射中沒有簡單數(shù)據(jù)類型秽梅,所有的編譯時類型都是對象。
反射把編譯時應該解決的問題留到了運行時剿牺。