鷹翔宇空

          學(xué)習(xí)和生活

          BlogJava 首頁 新隨筆 聯(lián)系 聚合 管理
            110 Posts :: 141 Stories :: 315 Comments :: 1 Trackbacks
          引自:http://www.aygfsteel.com/flyingis/archive/2005/11/09/18959.html

          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

          posted on 2006-01-13 12:38 TrampEagle 閱讀(191) 評(píng)論(0)  編輯  收藏 所屬分類: java
          主站蜘蛛池模板: 金沙县| 延吉市| 库车县| 屯昌县| 嵩明县| 和林格尔县| 静安区| 安西县| 邹平县| 乃东县| 叶城县| 博湖县| 延川县| 卓资县| 梅河口市| 田阳县| 平塘县| 图木舒克市| 丰城市| 万安县| 丰镇市| 通河县| 搜索| 夹江县| 无棣县| 出国| 定兴县| 德清县| 宁夏| 图木舒克市| 且末县| 新津县| 沁水县| 天等县| 黔东| 三穗县| 桃园市| 涿州市| 柳林县| 安福县| 长岛县|