引自:
http://www.tkk7.com/flyingis/archive/2005/11/09/18959.html3. 使用XPath語法來查詢對象和集合
Commons JXPath是一種讓人很吃驚地(非標準的)對XML標準的使用。XPath一段時間以來一直是作為在一個XSL樣式表中選擇結點或結點集的一種方法。如果你用過XML,你會很熟悉用這樣的語法/foo/bar來從foo文檔元素中選擇bar子元素。
Jakarta Commons JXPath增加了一種有趣的手法:你可以用JXPath來從bean和集合中選擇對象,其中如servlet上下文和DOM文檔對象??紤]一個包含了Person對象的列表。每一個Person對象有一個屬性的類型為Job,每一個Job對象有一個salary(薪水)屬性,類型為int。Person對象也有一個coountry屬性,它是兩個字符的國家代碼。使用JXPath,你可以很容易地選出所有國家為美國,薪水超過一百萬美元的Person對象。下面是設置一個由JXPath過濾地bean的List的代碼:
// Person的構造器設置姓和國家代碼
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的構造器設工作名稱和薪水
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包含了四個bean: Tim, John, Al, 和George。Tim是一個掙4萬美元的開發者,John是一個掙15萬美元的參議員,Al是一個掙340萬美元的喜劇演員,Tony是一個掙200萬歐元的部長。我們的任務很簡單:遍歷這個List,打印出每一個掙錢超過100百萬美元的美國公民的名字。記住people是一個由Person對象構成的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() );
??????}
????}
??}
}
上面的例子是繁重的,并有些容易犯錯。為了發現合適的Person對象,你必須首先遍歷每一個Person對象并且檢查conuntry的屬性。如果country屬性不為空并且符合要求,那么你就要檢查job屬性并看一下它是否不為空并且salary屬性的值大于100萬。上面的例子的代碼行數可以被Java 1.5的語法大大減少,但是,哪怕是Java 1.5,你仍舊需要在兩層上作兩次比較。
如果你想對內存中的一組Person對象也做一些這樣的查詢呢?如果你的應用想顯示所有在英格蘭的名叫Tony的人呢?喔,如果你打印出每一個薪水少于2萬的工作的名稱呢?
如果你將這些對象存儲到關系數據庫中,你可以用一個SQL查詢來解決問題,但你正在處理的是內存中的對象,你可以不必那么奢侈。雖然XPath主要是用在XML上面,但你可以用它來寫一個針對對象集合的“查詢”,將對象作為元素和,把bean屬性作為子元素。是的,這是一種對XPath奇怪的應用,但請先看一下下面的例子如何在people上,一個由Person對象構成的ArrayList,實現這三種查詢:
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()方法使用了一個XPath表達式,將它應用到一個集合上。XPath表達式被JXPathContext求值, JXPathContext由JXPathContext.newContext()調用創建,并將它傳入要執行查詢的集合中。凋用context.iterate()來在集合中的每一個元素上應用XPath表達式,返回包含所有符合條件的“節點”(這里是“對象”)的Iterator。上例中執行的第一個查詢,query1,執行了和不使用JXPath的例子相同的查詢。query2選擇所有國家為GB并且名字屬性為Tony的Person對象,query3返回了一個String對象的List,包含了所有Job對象的name屬性。
當我第一次看到Commons JXPath, 它是一個壞思想的想法觸動了我。為什么要把XPath表達式應用到對象上?有點感覺不對。把XPath作為一個bean的集合的查詢語言的這種意想不到的用法,在過去幾年中已經好多次給我帶來了便利。如果你發現你在list中循環來查找符合條件的元素,請考慮一下JXPath。更多的信息,請參考Jakarta Commons Cookbook的第12章,“查找和過濾”,它討論了Commons JXPath和與Commons Digester配對的Jakarta Lucene。
還有更多
對Jakarta Commons縱深地探索仍然在調試中。在這一系列的下面幾部分中,我會介紹一些相關的工具和功能。在Commons Collections中設置操作,在collection中使用Predicate對象,使用Commons Configuration來配置一個應用和使用Commons Betwixt來讀寫XML。能從Jakarta Commons得到的東西還有很多,不能在幾千字中表達,所以我建議你看一下Jakarta Commons Cookbook。許多功能可能會,一眼看上去,有點普通,但Jakarta Commons的能量就蘊藏在這些工具的相互組合和與你的系統的集成當中。
Timothy M. O'Brien是一個專業的獨立的開發者,在Chicago地區工作和生活。
資源
·onjava.com:onjava.com
·Matrix-Java開發者社區:http://www.matrix.org.cn/
·APACHE:APACHE.org