鷹翔宇空

          學習和生活

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

          3. 使用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

          posted on 2006-01-13 12:38 TrampEagle 閱讀(188) 評論(0)  編輯  收藏 所屬分類: java
          主站蜘蛛池模板: 崇文区| 武鸣县| 徐闻县| 漳浦县| 大名县| 漠河县| 会宁县| 长治县| 田林县| 双牌县| 理塘县| 响水县| 繁峙县| 固原市| 察雅县| 东乡县| 伊金霍洛旗| 桦川县| 临高县| 霍邱县| 马关县| 城步| 平舆县| 集贤县| 合江县| 兴安盟| 洛宁县| 砚山县| 廊坊市| 建瓯市| 浮梁县| 墨竹工卡县| 舒兰市| 无为县| 株洲县| 灌云县| 南康市| 西盟| 金沙县| 咸宁市| 呼和浩特市|