Java語言程序設計 課件 第11、12章 記錄、枚舉和注解類型;泛型與集合_第1頁
Java語言程序設計 課件 第11、12章 記錄、枚舉和注解類型;泛型與集合_第2頁
Java語言程序設計 課件 第11、12章 記錄、枚舉和注解類型;泛型與集合_第3頁
Java語言程序設計 課件 第11、12章 記錄、枚舉和注解類型;泛型與集合_第4頁
Java語言程序設計 課件 第11、12章 記錄、枚舉和注解類型;泛型與集合_第5頁
已閱讀5頁,還剩153頁未讀, 繼續(xù)免費閱讀

下載本文檔

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

文檔簡介

Java語言程序設計第11章記錄、枚舉和注解類型

1記錄類型23主要內(nèi)容Java語言程序設計(第4版)

2022枚舉類型注解類型Java語言程序設計11.1記錄類型在Java程序開發(fā)中,我們經(jīng)常需要定義一些數(shù)據(jù)類,比如與數(shù)據(jù)庫表交互的JavaBeans類或POJO類,這些類通常使用字段表示數(shù)據(jù),然后我們要為該類定義構造方法,為字段定義訪問方法和修改方法。這就需要程序員編寫大量的代碼。記錄類型從Java16開始,如果要定義這樣的類,可以使用record關鍵字將它定義為一個記錄類型。這種數(shù)據(jù)類型提供了一種緊湊的語法來聲明一種主要用于保存數(shù)據(jù)的類。假設定義一個Customer記錄類型,它帶兩個字段name和address,那么該類可能需要如下定義。packagecom.boda.xy;publicrecordCustomer(Stringname,Stringaddress){//這里可以定義記錄類型的成員}這里,類型名后面是一對括號,里面是字段的聲明,這相當于定義一個構造方法。記錄類型與其他類型一樣被編譯成類(.class)文件。對于記錄類型,編譯器將自動添加構造方法、equals()方法、hashCode()方法和toString()方法,并且為每個實例變量添加訪問方法(但不提供修改方法)。程序11.1Customer.java如果我們反編譯Customer.class文件,可得到類似下面的代碼。publicfinalclassCustomerextendsjava.lang.Record{privatefinaljava.lang.Stringname;privatefinaljava.lang.Stringaddress;publicCustomer(java.lang.Stringname,java.lang.Stringaddress){/*編譯的代碼*/}publicjava.lang.Stringname()

{/*編譯的代碼*/}publicjava.lang.Stringaddress()

{/*編譯的代碼*/}publicfinaljava.lang.StringtoString()

{/*編譯的代碼*/}publicfinalinthashCode()

{/*編譯的代碼*/}publicfinalbooleanequals(java.lang.Objecto)

{/*編譯的代碼*/}可以看到,聲明的Customer類繼承了java.lang.Record類,Customer類和它的兩個成員被聲明為final,也就是記錄類型不能被繼承,成員也不能被修改。記錄類型編譯器自動為它添加構造方法、equals()方法、hashCode()方法和toString()方法,并且為實例變量添加訪問方法。注意,這里的訪問方法名為實例名,如name()、address()等,而不是getName()這種形式的。publicclassCustomerDemo{publicstaticvoidmain(String[]args){varcustomer=newCustomer("張明月","北京市海淀區(qū)");varcustomer2=newCustomer("李大海","上海市科技路20號");System.out.println("姓名:"+customer.name());System.out.println("地址:"+customer.address());

System.out.println(customer.toString());System.out.println(customer.equals(customer2));System.out.println(customer.hashCode());System.out.println(customer2.hashCode());}}程序11.2CustomerDemo.java在記錄的主體中,還可以聲明static成員、構造方法和實例方法,例如:publicCustomer(Stringname){this(name,null);}記錄類型publicstaticStringinfo="客戶";publicstaticvoidshow(){

System.out.println("顯示:"+info);}自定義構造方法必須明確調用帶參數(shù)構造方法自定義靜態(tài)變量和靜態(tài)方法publicvoidshowName(){System.out.println("姓名:"+name);}記錄類型【注意】

不能在記錄類型中聲明實例變量,但可以聲明靜態(tài)變量。【注意】

記錄類型字段的訪問器名是“屬性名()”,而不是getXxx(),例如,假設記錄類型有一個name屬性,則它的訪問器為name(),而不是getName()。此外,記錄類型沒有修改方法,也就是沒有類似的setName()方法。實例方法在記錄體中還可以覆蓋超類Record中定義的方法,下面代碼覆蓋了toString()方法和hashCode()方法。記錄類型@OverridepublicStringtoString(){ return"姓名:"+name+",地址:"+address;} @OverridepublicinthashCode(){ returnObjects.hash(name,address);}在記錄類型中用戶還可以定義自己的方法和構造方法,但通常不這樣做。記錄類型主要是解決用于存儲數(shù)據(jù)的普通類的一個常見問題??偨Y下面對記錄類型做一簡單總結:記錄類型默認繼承了java.lang.Record類,不能顯式繼承其他類。記錄類型是final的,即它不可以被繼承。每個成員變量都被加上privatefinal,對象創(chuàng)建后它們就不可變。每個成員變量都提供了public訪問方法,如name(),但不提供修改方法。提供了帶所有參數(shù)的構造方法、toString()方法、equals()方法和hashCode()方法。1.定義一個名為Book的記錄類型,要求它實現(xiàn)Serializable接口和Comparable接口,Book記錄包含5個字段,如下所示。idint,nameString,authorString,pricedouble,pressString編程練習Java語言程序設計11.2枚舉類型在實際應用中,有些數(shù)據(jù)的取值被限定在幾個確定的值之內(nèi)。例如,一年有4個季度,一周有7天、一副紙牌有4種花色等。對這種類型的數(shù)據(jù)可以定義為枚舉類型。11.2.1枚舉類型的定義枚舉類型是一種特殊的引用類型,它的聲明和使用與類和接口有類似的地方。枚舉類型的聲明使用enum關鍵字。下面程序定義了一個名為Direction的枚舉類型:

publicenumDirection{EAST,SOUTH,WEST,NORTH;}編譯后產(chǎn)生一個Direction.class類文件枚舉類型都隱含地繼承了java.lang.Enum抽象類,Enum類又是Object類的子類,同時實現(xiàn)了Comparable接口。每個枚舉類型都包含了若干方法,下面是一些常用的。

staticE[]values()

staticEvalueOf(Stringname)

finalintcompareTo(Eo)

finalStringname()

finalintordinal()11.2.2枚舉類型的方法publicstaticvoidmain(String[]args){//聲明一個枚舉類型變量,并用一個枚舉賦值varleft=Direction.WEST;System.out.println(left);//輸出:WEST//輸出每個枚舉對象及序號for(Directiond:Direction.values()){System.out.println(()+",序號"+d.ordinal());}}程序11.4DirectionDemo.java枚舉類型有一個特別實用的特性,它可以在switch語句中使用。java.time.DayOfWeek是一個枚舉類型,其中包括一周的7天,分別為MONDAY、TUESDAY、WEDNESDAY、THURSDAY、FRIDAY、SATURDAY和SUNDAY,序號從0到6。EnumSwitch.java程序在switch結構中使用DayOfWeek枚舉。11.2.3枚舉在switch中的應用publicstaticvoiddescribe(DayOfWeekday){switch(day){caseMONDAY->System.out.println("Mondaysarebad.");caseFRIDAY->System.out.println("Fridaysarebetter.");caseSATURDAY,SUNDAY->System.out.println("Weekendsarebest.");default->System.out.println("Midweekdaysareso-so.");}}程序11.5EnumSwitch.javapublicstaticvoidmain(String[]args){varfirstDay=DayOfWeek.MONDAY;describe(firstDay);varthirdDay=DayOfWeek.WEDNESDAY;describe(thirdDay);varseventhDay=DayOfWeek.SUNDAY;describe(seventhDay);}在枚舉類型的聲明中,除了枚舉常量外還可以聲明構造方法,成員變量和其他方法。下面程序定義了Color枚舉,它包含4種顏色。11.2.4枚舉類型的構造方法publicenumColor{

RED("紅色",1),GREEN("綠色",2),WHITE("白色",3),YELLOW("黃色",4);

//成員變量privateStringname;privateintindex;

//構造方法privateColor(Stringname,intindex){=name;this.index=index;

}程序11.6Color.java

//普通方法publicstaticStringgetName(intindex){for(Colorc:Color.values()){if(c.getIndex()==index){return;

}

}returnnull;

}//getter和setter方法publicStringgetName(){returnname;

}publicvoidsetName(Stringname){=name;

}publicintgetIndex(){returnindex;

}publicvoidsetIndex(intindex){this.index=index;

}

//覆蓋方法@OverridepublicStringtoString(){return+"_"+this.index;

}publicstaticvoidmain(String[]args){Colorc=Color.RED;//自動調用構造方法System.out.println(c.toString());//輸出:1-紅色

}Java語言程序設計11.3注解類型注解以結構化的方式為程序元素提供信息,這些信息能夠被外部工具(編譯器、解釋器等)自動處理。注解類型注解有許多用途,其中包括:

為編譯器提供信息。

編譯時或部署時處理。

運行時處理。區(qū)分注解類型和注解注解類型是一種特殊的接口類型,像使用類一樣,要使用注解必須先定義注解類型(也可以使用語言本身提供的注解類型)。11.3.1注解概述注解是注解類型的一個實例。就像接口一樣,注解類型也有名稱和成員。注解中包含的信息采用“鍵/值”對的形式,可以有零或多個“鍵/值”對,并且每個鍵有一個特定類型。它可以是一個Stirng、int或其他Java類型。沒有“鍵/值”對的注解類型稱作標記注解類型。如果注解只需要一個“鍵/值”對,則稱為單值注解類型。11.3.1注解概述在Java程序中為程序元素標注注解的語法如下:

@AnnotationType或者

@AnnotationType(elementValuePairs)對沒有默認值的元素,都應該以name=value的形式對元素初始化。如果注解類型是標記注解類型(無元素),或者所有的元素都具有默認值,那么就可以省略初始化器列表。11.3.1注解概述如果注解類型只有一個元素,可以使用縮略的形式對注解元素初始化。例如,假設注解類型Copyright只有一個String類型的元素,用它注解程序元素時就可以寫作:

@Copyright("copyright2010-2015")11.3.1注解概述可以給Java包、類型(類、接口、枚舉)、構造方法、方法、成員變量、參數(shù)及局部變量進行標注。例如,可以給一個Java類進行標注,以便阻止javac程序可能發(fā)出的警告,也可以對一個要覆蓋的方法進行標注,讓編譯器知道你是要覆蓋這個方法而不是重載它。這里介紹Java語言規(guī)范中定義的3個注解類型,它們是供編譯器使用的。它們定義在java.lang包中,分別為@Override@Deprecated@SuppressWarnings11.3.2標準注解@Override是一個標記注解類型,可以用在一個方法的聲明中,它告訴編譯器這個方法要覆蓋父類中的某個方法。使用該注解可以防止程序員在覆蓋某個方法時出錯。例如,考慮下面的Parent類:

classParent{

publicdoublecalculate(doublex,doubley){

returnx*y;

}}假設現(xiàn)在要擴展Parent類,并覆蓋它的calculate()方法。下面是Parent類的一個子類:

classChildextendsParent{

publicintcalculate(intx,inty){

return(x+1)*y;

}}Child類可以編譯。然而,Child類中的calculate()方法并沒有覆蓋Parent中的方法,因為它的參數(shù)是2個int型,而不是2個double型。使用Override注解就可以很容易防止這類錯誤。每當你想要覆蓋一個方法時,就在這個方法前聲明Override注解類型:classChildextendsParent{

@Override

publicintcalculate(intx,inty){

return(x+1)*y;

}}@Deprecated是一個標記注解類型,可以應用于某個方法或某個類型,指明方法或類型已被棄用。警告代碼用戶不應該使用或者覆蓋該方法,或者不該使用或擴展該類型。一個方法或類型被標記棄用通常是因為有了更好的方法或類型。2.@Deprecated注解下面代碼使用了Deprecated注解。@DeprecatedpublicvoidbadMethod(){

System.out.println("Deprecated");

}

publicstaticvoidmain(String[]args){

DeprecatedDemo

dd=newDeprecatedDemo();

dd.badMethod();}在Eclipse中棄用的方法加刪除線調用被棄用的方法也加刪除線@SuppressWarnings注解指示編譯器阻止某些類型的警告,具體警告類型可以用初始化該注解的字符串來定義。該注解可應用于類型、構造方法、方法、成員變量、參數(shù)以及局部變量。它的用法是傳遞一個String數(shù)組,其中包含需要阻止的警告。語法如下:

SuppressWarnings(value={string-1,…,string-n})3.@SuppressWarnings注解以下是SuppressWarnings注解的常用有效參數(shù):unchecked,未檢查的轉換警告。deprecation,使用了不推薦使用的方法的警告。serial,實現(xiàn)Serializable接口但沒有定義serialVersionUID常量的警告。rawtypes,如果使用舊的語法創(chuàng)建泛型類對象時發(fā)出的警告。finally,任何finally子句不能正常完成的警告。fallthrough,switch塊中某個case后沒有break語句的警告。3.@SuppressWarnings注解下面程序阻止了代碼中出現(xiàn)的幾種編譯警告。@SuppressWarnings(value={"unchecked","serial","deprecation"})publicclassSuppressWarningDemoimplementsSerializable{publicstaticvoidmain(String[]args){Dated=newDate();

System.out.println(d.getDay());ListmyList=newArrayList();//該語句仍然有警告

myList.add("one");

myList.add("two");

myList.add("three");

System.out.println(myList);}}程序11.7SuppressWarningDemo.java用戶也可以定義注解類型。注解類型的定義與接口類型的定義類似。注解類型的定義使用interface關鍵字,前面加上@符號。

public@interfaceCustomAnnotation{

//元素或屬性聲明

}在注解類型中聲明的方法稱為注解類型的元素,它的聲明類似于接口中的方法聲明,沒有方法體,但有返回類型。11.3.3定義注解類型元素的類型有一些限制,如只能是基本類型、String、枚舉類型、其他注解類型等,并且元素不能聲明任何參數(shù)。在定義注解時可以使用default關鍵字為元素指定默認值。例如:

public@interfaceVersion{

intmajor()default1;

intminor()default0;}11.3.3定義注解類型Version注解類型可以用來標注類和接口,也可以供其他注解類型使用。例如,可以用它來重新定義ClassInfo注解類型:

public@interfaceClassInfo{Stringcreated();Stringauthor();StringlastModified();

Versionversion();}11.3.3定義注解類型注解類型中也可以沒有元素,這樣的注解稱為標記注解(markerannotation),這與標記接口類似。例如,下面定義了一個標記注解類型Preliminary:public@interfacePreliminary{}11.3.3定義注解類型如果注解類型只有一個元素,這個元素應該命名為value。例如,Copyright注解類型只有一個String類型的元素,則其應該定義為:

public@interfaceCopyright{Stringvalue();}這樣,在為程序元素注解時就可以不需要指定元素名稱,而采用一種縮略的形式:

@Copyright("flyingdragoncompany")11.3.3定義注解類型1.定義一個名為Product記錄類型表示商品信息,它的UML圖如圖11-1所示。要求為該記錄類型定義一個僅帶id參數(shù)的構造方法,覆蓋父類的equals()方法和hashCode()方法,為保證hashCode()方法和equals()方法兼容的,請使用Objects類的hash()方法和equals()。覆蓋父類的toString()方法,要求當調用該方法是輸出Product各屬性信息。編寫程序測試Product記錄類型所有方法的使用。編程作業(yè)2.

定義一個名為TrafficLight的enum類型,它包含三個常量:GREEN、RED和YELLOW表示交通燈的三種顏色。通過values()方法和ordinal()方法循環(huán)并打印每一個值及其順序值。編寫一個switch語句,為TrafficLight的每個常量輸出有關信息。編程作業(yè)Java語言程序設計第12章泛型與集合

1泛型23主要內(nèi)容Java語言程序設計(第4版)

2022集合框架List接口及實現(xiàn)類45Set接口及實現(xiàn)類Queue接口及實現(xiàn)類6案例:用集合存儲、遍歷員工記錄7Map接口及實現(xiàn)類89Collections類案例:用Map統(tǒng)計單詞數(shù)量Java語言程序設計12.1泛型泛型是類和接口的一種擴展機制,主要實現(xiàn)參數(shù)化類型機制。使用泛型,程序員可以編寫更安全的程序。概述泛型是Java5引進的新特征,泛型被廣泛應用在Java集合API中,在Java集合框架中大多數(shù)的類和接口都是泛型類型?;仡櫼幌?,我們在8.5節(jié)中定義了一個整數(shù)棧類IntStack,該類使用Integer作為棧的元素,這就限制了該類只能對Integer元素操作。12.1.1泛型類如果要使這個棧類更具有通用性,我們可以使用Object作為棧的元素,因為Object類是所有類的超類,所以Object可以引用任何對象類型。然而,這種做法無法提供類型的安全性,在進行類型轉換時可能發(fā)生類型不匹配異常。使用泛型就可以提高類型安全性,因為,它可以使類型轉換自動地、隱式地進行。所謂泛型(generics)就是帶一個或多個類型參數(shù)(typeparameter)的類或接口。對于上述討論的對象棧,可以使用泛型定義。12.1.1泛型類為了簡單,下面先定義一個泛型Node類表示節(jié)點,類型參數(shù)T表示節(jié)點中存放的數(shù)據(jù)值類型。具體代碼如程序12.1所示。publicclassNode<T>{privateTdata;//泛型成員,T可以是任何引用類型publicNode(){}//默認構造方法publicNode(Tdata){//帶參數(shù)構造方法this.data=data;}publicTget(){//訪問方法定義returndata;}程序12.1Node.javapublicvoidset(Tdata){//修改方法定義this.data=data;}//顯示類型名publicvoidshowType(){ System.out.println("T的類型是:"+data.getClass().getName());}}泛型類型的使用與方法調用類似,方法調用需向方法傳遞參數(shù),使用泛型需傳遞一個類型參數(shù),即用某個具體的類型替換T。例如,如果要在Node對象中存放Integer對象,就需要在創(chuàng)建Node對象時為其傳遞Integer類型參數(shù)。12.1.1泛型類要實例化泛型類對象,也使用new運算符,但在類名后面需加上要傳遞的具體類型。Node<Integer>

intNode=newNode<Integer>();一旦創(chuàng)建了intNode對象,就可以調用set()方法設置其中的Integer對象,如下代碼所示。

publicstaticvoidmain(String[]args){Node<Integer>intNode=newNode<Integer>();intNode.set(Integer.valueOf(999));System.out.println(intNode.get());intNode.showType();}由于編譯器能夠從上下文中推斷出泛型參數(shù)的類型,所以從JavaSE7開始,在創(chuàng)建泛型類型時可使用菱形語法(diamond),即僅用一對尖括號(<>),上述創(chuàng)建intNode的語句可以寫成:Node<Integer>intNode=newNode<>();12.1.1泛型類按照約定,類型參數(shù)名使用單個大寫字母表示。常用的類型參數(shù)名有:E表示元素,K表示鍵,N表示數(shù)字,T表示類型,V表示值等。需要注意,泛型可能具有多個類型參數(shù),但在類或接口的聲明中,每個參數(shù)名必須是唯一的。與類一樣,接口也可以聲明為泛型接口,即接口也可以帶參數(shù),例如,下面的Readable接口使用泛型類型作為其read()方法的參數(shù):publicinterfaceReadble<T>{voidread(Tt);}12.1.2泛型接口類實現(xiàn)這個接口有三種方法。

第一個是指定類中的泛型類型。下面的具體類說它只處理Book。這使得它可以用一個Book參數(shù)聲明read()方法:publicclassIntelligentRobotimplementsReadable<Book>{publicvoidread(Bookb){}}12.1.2泛型接口下一種方法是創(chuàng)建一個泛型類。下面的具體類允許調用者指定泛型的類型:publicclassIntelligentRobot<U>implementsReadable<U>{publicvoidread(Ut){}}在本例中,類型參數(shù)可以被命名為任何名稱,包括T。我們在例子中使用了U,這樣就不會混淆T指的是什么。12.1.2泛型接口最后一種方法是不使用泛型。這是編寫代碼的老方法。它產(chǎn)生一個編譯器警告Readable是一個原始類型,但它能夠編譯。這里的read()方法有一個Object參數(shù),因為泛型類型沒有定義:publicclassIntelligentRobotimplementsReadable{publicvoidread(Objecto){}}12.1.2泛型接口泛型接口也可帶有多個類型參數(shù)。下面的Entry是泛型接口,帶兩個類型參數(shù)。12.1.2泛型接口publicinterfaceEntry<K,V>{publicKgetKey();publicVgetValue();}publicclassPair<K,V>implementsEntry<K,V>{privateKkey;privateVvalue;publicPair(Kkey,Vvalue){//構造方法this.key=key;this.value=value;}程序12.2Pair.javapublicvoidsetKey(Kkey){this.key=key;}publicKgetKey(){returnkey;}publicvoidsetValue(Vvalue){this.value=value;}publicVgetValue(){returnvalue;}}下面語句創(chuàng)建兩個Pair類實例:Pair<String,Integer>p1=newPair<>("twenty-two",22);Pair<String,String>p2=newPair<>("china","Beijing");泛型方法(genericmethod)是帶類型參數(shù)的方法。類的成員方法和構造方法都可以定義為泛型方法。泛型方法的定義與泛型類型的定義類似,但類型參數(shù)的作用域僅限于聲明的方法和構造方法內(nèi)。泛型方法可以定義為靜態(tài)的和非靜態(tài)的。12.1.3泛型方法下面的MathUtils類中定義兩個static的泛型方法swap()和compare()。swap()方法用于交換任何數(shù)組中兩個元素(數(shù)組元素類型不是基本類型),compare()方法用于比較兩個泛型類Pair對象的參數(shù)K和V是否相等。特別注意,對于泛型方法必須在方法返回值前指定泛型,如<K,V>。publicclassMathUtils{publicstatic<T>voidswap(T[]array,inti,intj){Ttemp=array[i];array[i]=array[j];array[j]=temp;}publicstatic<K,V>booleancompare(Pair<K,V>p1,Pair<K,V>p2){returnp1.getKey().equals(p2.getKey())&&p1.getValue().equals(p2.getValue());}程序12.3MathUtils.javapublicstaticvoidmain(String[]args){Integer[]numbers={1,3,5,7};MathUtils.swap(numbers,0,3);for(Integern:numbers){System.out.println(n+"");//輸出:7351}Pair<Integer,String>p1=newPair<>(1,"apple");Pair<Integer,String>p2=newPair<>(2,"orange");//調用泛型方法booleansame=MathUpare(p1,p2);System.out.println(same);//輸出:false}泛型類型本身是一個Java類型,就像java.lang.String和java.time.LocalDate一樣,為泛型類型傳遞不同的類型參數(shù)會產(chǎn)生不同的類型。例如,下面list1和list2就是不同的類型對象。

List<Object>list1=newArrayList<Object>();List<String>list2=newArrayList<String>();12.1.4通配符(?)的使用這里List和ArrayList是泛型接口和泛型類。盡管String是Object的子類,但List<String>與List<Object>卻沒有關系,List<String>并不是List<Object>的子類型。因此,把一個List<String>對象傳遞給一個需要List<Object>對象的方法,將會產(chǎn)生一個編譯錯誤。請看下面代碼。12.1.4通配符(?)的使用publicstaticvoidprintList(List<Object>list){for(Objectelement:list){

System.out.println(element);}}該方法的功能是打印傳遞給它的列表的所有元素。如果傳遞給該方法一個List<String>對象,將發(fā)生編譯錯誤。如果要使上述方法可打印任何類型的列表,可將其參數(shù)類型修改為List<?>,如下所示:publicstaticvoidprintList(List<?>list){for(Objectelement:list){

System.out.println(element);}}這里,問號(?)就是通配符,它表示該方法可接受元素是任何類型的List對象。12.1.4通配符(?)的使用publicclassWildCardDemo{publicstaticvoidprintList(List<?>list){ for(Objectelement:list){ System.out.println(element); }}程序12.4WildCardDemo.javapublicstaticvoidmain(String[]args){ varmyList=newArrayList<String>(); myList.add("cat"); myList.add("dog"); myList.add("horse");printList(myList);}有時需要限制傳遞給類型參數(shù)的類型種類,例如,要求一個方法只接受Number類或其子類的實例,這就需要使用有界類型參數(shù)(boundedtypeparameter)。12.1.5有界類型參數(shù)有界類型分為上界和下界,上界用extends指定,下界用super指定。上界用extends指定,例如,List<?extendsNumber>,Number或Number的子類型下界用super指定,例如,List<?superInteger>,Integer或Integer的超類型publicstaticdoublegetAverage(List<?extendsNumber>numberList){ vartotal=0.0; for(varnumber:numberList){ total+=number.doubleValue(); } returntotal/numberList.size();}程序12.5BoundedTypeDemo.javapublicstaticvoidmain(String[]args){

varintegerList=newArrayList<Integer>(); integerList.add(3); integerList.add(30); integerList.add(300); System.out.println(getAverage(integerList));//輸出:111.0

vardoubleList=newArrayList<Double>(); doubleList.add(5.5); doubleList.add(55.5); System.out.println(getAverage(doubleList));//輸出:30.5當實例化泛型類型時,編譯器使用一種叫類型擦除(typeerasure)的技術轉換這些類型。在編譯時,編譯器將清除類和方法中所有與類型參數(shù)有關的信息。類型擦除可讓使用泛型的Java應用程序與之前不使用泛型類型的Java類庫和應用程序兼容。2.1.6類型擦除例如,Node<Integer>被轉換成Node,它稱為源類型(rawtype)。源類型是不帶任何類型參數(shù)的泛型類或接口名。這說明在運行時找不到泛型類使用的是什么類型。下面的操作是不可能的。publicclassMyClass<E>{publicstaticvoidmyMethod(Objectitem){if(iteminstanceofE){//編譯錯誤…}

Eitem2=newE();//編譯錯誤E[]iArray=newE[10];//編譯錯誤Eobj=(E)newObject();//非檢查的造型警告}}Java語言程序設計12.2集合框架在編寫面向對象的程序時,經(jīng)常要用到一組類型相同的對象??梢允褂脭?shù)組來集中存放這些類型相同的對象,但數(shù)組一經(jīng)定義便不能改變大小。因此,Java提供了一個集合框架(collectionsframework),該框架定義了一組接口和類,使處理對象組更容易。集合框架集合是指集中存放一組對象的一個對象。集合相當于一個容器。集合能夠幫助Java程序員輕松地管理對象。Java集合框架由兩種類型構成,一個是Collection,另一個是Map。Collection對象用于存放一組對象,Map對象用于存放一組“關鍵字/值”的對象。Collection<E>接口是所有集合類型的根接口,它繼承了Iterable<E>接口,它有三個子接口:Set接口、List接口和Queue接口。Collection接口定義了集合操作的常用方法。可以簡單分為以下幾類:基本操作、批量操作、數(shù)組操作和流操作。集合框架實現(xiàn)基本操作的方法添加元素、刪除指定元素、返回集合中元素的個數(shù)、返回集合的迭代器對象。1.基本操作booleanadd(Ee):向集合中添加元素e。booleanremove(Objecto):從集合中刪除指定的元素o。booleancontains(Objecto):返回集合中是否包含指定的元素o。booleanisEmpty():返回集合是否為空,即不包含元素。intsize():返回集合中包含的元素個數(shù)。Iteratoriterator():返回包含所有元素的迭代器對象。下面的方法可實現(xiàn)集合的批量操作。booleanaddAll(Collection<?extendsE>c):將集合c中的所有元素添加到當前集合中。booleanremoveAll(Collection<?>c):從當前集合中刪除c中的所有元素。defaultbooleanremoveIf(Predicate<?superE>filter)booleancontainsAll(Collection<?>c):返回當前集合是否包含c中的所有元素。booleanretainAll(Collection<?>c):在當前集合中只保留指定集合c中的的元素,其他元素刪除。

voidclear():將集合清空。

2.批量操作下面方法可以將集合元素轉換成數(shù)組元素。Object[]toArray():返回包含集合中所有元素的對象數(shù)組。

<T>T[]toArray(T[]a):返回包含集合中所有元素的數(shù)組,返回數(shù)組的元素類型是指定的數(shù)組類型。3.數(shù)組操作設c是一個Collection對象,下面的代碼將c中的對象轉換成一個新的Object數(shù)組,數(shù)組的長度與集合c中的元素個數(shù)相同。Object[]a=c.toArray();Java語言程序設計12.3List接口及實現(xiàn)類List接口是Collection的子接口,它實現(xiàn)一種線性表的數(shù)據(jù)結構。存放在

List中的所有元素都有一個下標(從0開始),可以通過下標訪問List中的元素。List中可以包含重復元素。List接口的實現(xiàn)類包括ArrayList、LinkedList、Vector和Stack。概述北京上海廣州bigCities012List接口除繼承Collection的方法外,還定義了一些自己的方法。使用這些方法可以實現(xiàn)定位訪問、查找、迭代和返回子線性表。List的常用方法如下。12.3.1List的操作booleanadd(Ee):向集合中添加元素e。voidadd(intindex,Eelement):將指定元素插入到指定下標處。Eget(intindex):返回指定下標處的元素。Eset(intindex,Eelement):修改指定下標處的元素。static

<E>

List<E>

of?(E...

elements):返回包含任意數(shù)量元素的不可修改列表。該方法還接受單個數(shù)組作為參數(shù)。結果列表的元素類型將是數(shù)組的元素類型,列表的大小等于數(shù)組的長度。Eremove(intindex):刪除指定下標處的元素。intindexOf(Objecto):查找指定對象第一次出現(xiàn)的位置。intlastIndexOf(Objecto):查找指定對象最后一次出現(xiàn)的位置。List<E>subList(intfrom,intto):返回從from到to元素的一個子線性表。12.3.1List的操作下面代碼使用List的of()方法創(chuàng)建一個不可修改的List對象,對不可修改的對象調用修改方法將拋出運行時異常。List<String>nameList=List.of("趙","錢","孫","李");System.out.println(nameList);//輸出:[趙,錢,孫,李]//nameList.add("周");//nameList.set(1,"吳");//nameList.remove("李");System.out.println(nameList.indexOf("孫"));//輸出:2產(chǎn)生運行時異常ArrayList類實際上實現(xiàn)了一個變長的對象數(shù)組,其元素可以動態(tài)地增加和刪除。它的定位訪問時間是常量時間。12.3.2ArrayList類ArrayList的構造方法如下:publicArrayList()public

ArrayList(Collectionc)public

ArrayList(intinitialCapacity)下列代碼創(chuàng)建一個ArrayList對象并向其中插入幾個元素,并使用ArrayList的有關方法對它操作。varcities=newArrayList<String>();cities.add("北京");cities.add("上海");cities.add("廣州");System.out.println(cities.size());cities.add(1,"倫敦");cities.set(1,"紐約");System.out.println(cities.contains("北京"));System.out.println(cities);System.out.println(cities.indexOf("巴黎"));遍歷集合中的元素有多種方法:用簡單的for循環(huán)、用增強的for循環(huán)和用Iterator迭代器對象。12.3.3遍歷集合元素1.使用for循環(huán)使用for循環(huán)可以遍歷集合中的每個元素。for(var

i=0;i<cities.size();i++){

System.out.print(cities.get(i)+"");}2.使用增強的for循環(huán)使用增強的for循環(huán)不但可以遍歷數(shù)組的每個元素,還可以遍歷集合中的每個元素。下面的代碼打印集合的每個元素:for(varcity:cities){

System.out.println(city);}12.3.3遍歷集合元素上述代碼的含義是:依次將集合cities中的每個對象存儲到city變量中,然后打印輸出。如果只是簡單輸出每個元素,可以調用集合對象的forEach()方法,給它傳遞一個System.out:println方法引用。

cities.forEach(System.out::println);12.3.3遍歷集合元素3.使用迭代器迭代器是一個可以遍歷集合中每個元素的對象。調用集合對象的iterator()方法可以得到Iterator對象,再調用Iterator對象的方法就可以遍歷集合中的每個元素。Iterator接口定義了如下3個方法。booleanhasNext():返回迭代器中是否還有對象。Enext():返回迭代器中下一個對象。voidremove():刪除迭代器中的當前對象。12.3.3遍歷集合元素Iterator使用一個內(nèi)部指針,開始它指向第一個元素的前面。如果在指針的后面還有元素,hasNext()方法返回true。調用next()方法,指針將移到下一個元素,并返回該元素。remove()方法將刪除指針所指的元素。假設myList是ArrayList的一個對象,要訪問myList中的每個元素,可以按下列方法實現(xiàn):Iteratoriterator=myList.iterator();//得到迭代器對象while(iterator.hasNext()){System.out.println(iterator.next());}使用Iterator也可以用for循環(huán)訪問集合元素。for(Iteratoriterator=myList.iterator();iterator.hasNext();){System.out.println(iterator.next());}4.雙向迭代器List還提供了listIterator()方法返回ListIterator對象。它可以從前后兩個方向遍歷線性表中元素,在迭代中修改元素以及獲得元素的當前位置。ListIterator是Iterator的子接口,它不但繼承了Iterator接口中的方法,還定義了自己的方法。12.3.3遍歷集合元素booleanhasPrevious():是否還有前一個元素。Eprevious():返回前一個元素。voidremove():刪除當前元素。voidset(Eo):修改當前元素。List<String>myList=List.of("北京","上海","廣州","深圳"); ListIterator<String>iterator=myList.listIterator();while(iterator.hasNext()){iterator.next();}//從后向前訪問列表每個元素while(iterator.hasPrevious()){ System.out.print(iterator.previous()+"");}程序12.8IteratorDemo.javajava.util.Arrays類提供了一個asList()方法,它實現(xiàn)將數(shù)組轉換成List對象的功能,該方法的定義如下。publicstatic<T>List<T>asList(T…a)該方法提供了一個方便的從多個元素創(chuàng)建List對象的途徑,它的功能與Collection接口的toArray()方法相反。12.3.4數(shù)組轉換為List對象String[]str={"one","two","three","four"};List<String>list=Arrays.asList(str);//將數(shù)組轉換為列表

System.out.println(list);Java語言程序設計12.4Set接口及實現(xiàn)類Set接口是Collection的子接口,Set接口對象類似于數(shù)學上的集合概念,其中不允許有重復的元素。Set接口沒有定義新的方法,只包含從Collection接口繼承的方法。Set接口的常用實現(xiàn)類有:HashSet類、TreeSet類和LinkedHashSet類。概述HashSet類用散列方法存儲元素,具有最好的存取性能,但元素沒有順序。HashSet類的構造方法有:12.4.1HashSet類HashSet類的構造方法有:HashSet()初始容量是16,默認裝填因子是0.75。HashSet(Collectionc)HashSet(int

initialCapacity)varwords=newHashSet<String>();words.add("one");words.add("two");words.add("three");words.add("one");for(varw:words){System.out.print(w+"");//輸出:onetwothree}使用Set對象的批量操作方法,可以實現(xiàn)標準集合代數(shù)運算。假設s1和s2是Set對象,下面的操作可實現(xiàn)相關的集合運算。12.4.2用Set對象實現(xiàn)集合運算s1.addAll(s2):實現(xiàn)集合s1與s2的并運算。s1.retainAll(s2):實現(xiàn)集合s1與s2的交運算。s1.removeAll(s2):實現(xiàn)集合s1與s2的差運算。s1.containAll(s2):如果s2是s1的子集,該方法返回true。設集合中存放的元素類型為Integer。為了計算兩個集合的并、交、差運算而又不破壞原來的集合,可以通過下面代碼實現(xiàn)。vars1=Set.of(1,2,3);vars2=Set.of(2,3,4);varunion=newHashSet<Integer>(s1);union.addAll(s2);//[1,2,3,4]varintersection=newHashSet<Integer>(s1);intersection.retainAll(s2);//[2,3]vardifference=newHashSet<Integer>(s1);difference.removeAll(s2);//[1]TreeSet實現(xiàn)一種樹集合,它使用紅-黑樹為元素排序,添加到TreeSet中的元素必須是可比較的,即元素的類必須實現(xiàn)Comparable<T>接口。它的操作要比HashSet慢。

TreeSet類的默認構造方法創(chuàng)建一個空的樹集合,其他構造方法如下。12.4.3TreeSet類TreeSet(Collectionc)TreeSet(Comparatorc)varts=newTreeSet<String>();//TreeSet中的元素將自動排序vars=newString[]{"one","two","three","four"};for(vari=0;i<s.length;i++){ts.add(s[i]);}System.out.println(ts);//輸出:[four,one,three,two]

從輸出結果中可以看到,這些字符串是按照自然順序排序的。程序12.9TreeSetDemo.java創(chuàng)建TreeSet類對象時如果沒有指定比較器對象,集合中的元素按自然順序排列。所謂自然順序(naturalorder)是指集合對象實現(xiàn)了Comparable<T>接口的compareTo()方法,對象則根據(jù)該方法排序。如果試圖對沒有實現(xiàn)Comparable<T>接口的集合元素排序,將拋出ClassCastException異常。12.4.4對象順序另一種排序方法是創(chuàng)建TreeSet對象時指定一個比較器對象,這樣,元素將按比較器的規(guī)則排序。如果需要指定新的比較規(guī)則,可以定義一個類實現(xiàn)Comparator<T>接口,然后為集合提供一個新的比較器。字符串的默認比較規(guī)則是按字母順序比較。假如按反順序比較,可以定義一個類實現(xiàn)Comparator<T>接口,然后用該類對象作為比較器。下面的程序就可以實現(xiàn)字符串的降序排序:12.4.4對象順序String[]s={"China","England","France","America","Russia",};varts=newTreeSet<String>();for(vari=0;i<s.length;i++){ts.add(s[i]);}System.out.println(ts);//按自然順序輸出程序12.10DescSortDemo.javats=newTreeSet<String>(

newComparator<String>(){

@Overridepublicintcompare(Stringa,Stringb){returnpareTo(a);}});

for(vari=0;i<s.length;i++){

ts.add(s[i]);}System.out.println(ts);創(chuàng)建一個比較器對象Java語言程序設計12.5Queue接口及實現(xiàn)類Queue接口是Collection的子接口,它是以先進先出(First-In-First-Out,F(xiàn)IFO)的方式排列其元素,一般稱為隊列(queue)。Deque接口是Queue接口的子接口,它的對象實現(xiàn)雙端隊列,ArrayDeque和LinkedList是它的兩個實現(xiàn)類。PriorityQueue實現(xiàn)的是一種優(yōu)先隊列,優(yōu)先隊列中元素的順序是根據(jù)元素的值排列的。概述Queue接口除了提供Collection的操作外,還提供了插入、刪除和檢查操作。12.5.1Queue接口和Deque接口Deque接口實現(xiàn)雙端隊列,它支持從兩端插入和刪除元素,它同時實現(xiàn)了Stack和Queue的功能。Deque接口中定義的基本操作方法,如表12-1所示。Deque接口常用方法操作類型隊首元素操作隊尾元素操作插入元素addFirst(e)offerFirst(e)addLast(e)offerLast(e)返回元素getFirst()peekFirst()getLast()peekLast()刪除元素removeFirst()pollFirstremoveLast()pollLast()Deque的常用實現(xiàn)類包括ArrayDeque類和LinkedList類,前者是

溫馨提示

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

最新文檔

評論

0/150

提交評論