3. 使用XPath語法來查詢對(duì)象和集合
Commons JXPath是一種讓人很吃驚地(非標(biāo)準(zhǔn)的)對(duì)XML標(biāo)準(zhǔn)的使用。XPath一段時(shí)間以來一直是作為在一個(gè)XSL樣式表中選擇結(jié)點(diǎn)或結(jié)點(diǎn)集的一種方法。如果你用過XML,你會(huì)很熟悉用這樣的語法/foo/bar來從foo文檔元素中選擇bar子元素。
Jakarta Commons JXPath增加了一種有趣的手法:你可以用JXPath來從bean和集合中選擇對(duì)象,其中如servlet上下文和DOM文檔對(duì)象。考慮一個(gè)包含了Person對(duì)象的列表。每一個(gè)Person對(duì)象有一個(gè)屬性的類型為Job,每一個(gè)Job對(duì)象有一個(gè)salary(薪水)屬性,類型為int。Person對(duì)象也有一個(gè)coountry屬性,它是兩個(gè)字符的國家代碼。使用JXPath,你可以很容易地選出所有國家為美國,薪水超過一百萬美元的Person對(duì)象。下面是設(shè)置一個(gè)由JXPath過濾地bean的List的代碼:
// Person的構(gòu)造器設(shè)置姓和國家代碼
Person person1 = new Person( "Tim", "US" );
Person person2 = new Person( "John", "US" );
Person person3 = new Person( "Al",??"US" );
Person person4 = new Person( "Tony", "GB" );
// Job的構(gòu)造器設(shè)工作名稱和薪水
person1.setJob( new Job( "Developer", 40000 ) );
person2.setJob( new Job( "Senator", 150000 ) );
person3.setJob( new Job( "Comedian", 3400302 ) );
person4.setJob( new Job( "Minister", 2000000 ) );
Person[] personArr =
??new Person[] { person1, person2,
???????????????? person3, person4 };
List people = Arrays.asList( personArr );
people List包含了四個(gè)bean: Tim, John, Al, 和George。Tim是一個(gè)掙4萬美元的開發(fā)者,John是一個(gè)掙15萬美元的參議員,Al是一個(gè)掙340萬美元的喜劇演員,Tony是一個(gè)掙200萬歐元的部長。我們的任務(wù)很簡(jiǎn)單:遍歷這個(gè)List,打印出每一個(gè)掙錢超過100百萬美元的美國公民的名字。記住people是一個(gè)由Person對(duì)象構(gòu)成的ArrayList,讓我們先看一下沒有利用JXPath便利的解決方案:
Iterator peopleIter = people.getIterator();
while( peopleIter.hasNext() ) {
??Person person = (Person) peopleIter.next();
??if( person.getCountry() != null &&
??????person.getCountry().equals( "US" ) &&
??????person.getJob() != null &&
??????person.getJob().getSalary() > 1000000 ) {
????????print( person.getFirstName() + " "
?????????????? person.getLastName() );
??????}
????}
??}
}
上面的例子是繁重的,并有些容易犯錯(cuò)。為了發(fā)現(xiàn)合適的Person對(duì)象,你必須首先遍歷每一個(gè)Person對(duì)象并且檢查conuntry的屬性。如果country屬性不為空并且符合要求,那么你就要檢查job屬性并看一下它是否不為空并且salary屬性的值大于100萬。上面的例子的代碼行數(shù)可以被Java 1.5的語法大大減少,但是,哪怕是Java 1.5,你仍舊需要在兩層上作兩次比較。
如果你想對(duì)內(nèi)存中的一組Person對(duì)象也做一些這樣的查詢呢?如果你的應(yīng)用想顯示所有在英格蘭的名叫Tony的人呢?喔,如果你打印出每一個(gè)薪水少于2萬的工作的名稱呢?
如果你將這些對(duì)象存儲(chǔ)到關(guān)系數(shù)據(jù)庫中,你可以用一個(gè)SQL查詢來解決問題,但你正在處理的是內(nèi)存中的對(duì)象,你可以不必那么奢侈。雖然XPath主要是用在XML上面,但你可以用它來寫一個(gè)針對(duì)對(duì)象集合的“查詢”,將對(duì)象作為元素和,把bean屬性作為子元素。是的,這是一種對(duì)XPath奇怪的應(yīng)用,但請(qǐng)先看一下下面的例子如何在people上,一個(gè)由Person對(duì)象構(gòu)成的ArrayList,實(shí)現(xiàn)這三種查詢:
import org.apache.commons.jxpath.JXPathContext;
public List queryCollection(String xpath,
????????????????????????????Collection col) {
????List results = new ArrayList();
????JXPathContext context =
????????JXPathContext.newContext( col );
????Iterator matching =
????????context.iterate( xpath );
????while( matching.hasNext() ) {
????????results.add( matching.getNext() );
????}
????return results;
}
String query1 =
?? ".[@country = 'US']/job[@salary > 1000000]/..";??
String query2 =
?? ".[@country = 'GB' and @name = 'Tony']";??
String query3 =
?? "./job/name";
List richUsPeople =
????queryCollection( query1, people );
List britishTony =
????queryCollection( query2, people );
List jobNames =
????queryCollection( query3, people );
queryCollection()方法使用了一個(gè)XPath表達(dá)式,將它應(yīng)用到一個(gè)集合上。XPath表達(dá)式被JXPathContext求值, JXPathContext由JXPathContext.newContext()調(diào)用創(chuàng)建,并將它傳入要執(zhí)行查詢的集合中。凋用context.iterate()來在集合中的每一個(gè)元素上應(yīng)用XPath表達(dá)式,返回包含所有符合條件的“節(jié)點(diǎn)”(這里是“對(duì)象”)的Iterator。上例中執(zhí)行的第一個(gè)查詢,query1,執(zhí)行了和不使用JXPath的例子相同的查詢。query2選擇所有國家為GB并且名字屬性為Tony的Person對(duì)象,query3返回了一個(gè)String對(duì)象的List,包含了所有Job對(duì)象的name屬性。
當(dāng)我第一次看到Commons JXPath, 它是一個(gè)壞思想的想法觸動(dòng)了我。為什么要把XPath表達(dá)式應(yīng)用到對(duì)象上?有點(diǎn)感覺不對(duì)。把XPath作為一個(gè)bean的集合的查詢語言的這種意想不到的用法,在過去幾年中已經(jīng)好多次給我?guī)砹吮憷H绻惆l(fā)現(xiàn)你在list中循環(huán)來查找符合條件的元素,請(qǐng)考慮一下JXPath。更多的信息,請(qǐng)參考Jakarta Commons Cookbook的第12章,“查找和過濾”,它討論了Commons JXPath和與Commons Digester配對(duì)的Jakarta Lucene。
還有更多
對(duì)Jakarta Commons縱深地探索仍然在調(diào)試中。在這一系列的下面幾部分中,我會(huì)介紹一些相關(guān)的工具和功能。在Commons Collections中設(shè)置操作,在collection中使用Predicate對(duì)象,使用Commons Configuration來配置一個(gè)應(yīng)用和使用Commons Betwixt來讀寫XML。能從Jakarta Commons得到的東西還有很多,不能在幾千字中表達(dá),所以我建議你看一下Jakarta Commons Cookbook。許多功能可能會(huì),一眼看上去,有點(diǎn)普通,但Jakarta Commons的能量就蘊(yùn)藏在這些工具的相互組合和與你的系統(tǒng)的集成當(dāng)中。
Timothy M. O'Brien是一個(gè)專業(yè)的獨(dú)立的開發(fā)者,在Chicago地區(qū)工作和生活。
資源
·onjava.com:onjava.com
·Matrix-Java開發(fā)者社區(qū):http://www.matrix.org.cn/
·APACHE:APACHE.org