使用jdom的XPath来快速定位xml节点

使用jdom的XPath来快速定位xml节点
Jagie 原创  (参与分:291,专家分:1420)   发表:2003-11-28 下午1:34   版本:1.0   阅读:3162

关键词:xpath,jdom



在用jdom来解析xml文档的时候,我们常常需要快速定位到某一个节点进行读取或修改操作。定位节点,这正是xpath表达式发挥作用的地方。从beta 9(www.jdom.org有下载)开始,jdom开始支持解析xpath路径表达式了,这样我们在用jdom来解析xml文档的时候,方便多了。请看例子

1.示例用xml文件,kk.xml

<?xml version = "1.0" encoding="GB2312" ?>
<联系人列表>
    <联系人>
        <姓名>张三</姓名>
        <ID>001</ID>
        <公司>A公司</公司>
        <EMAIL>[email protected]</EMAIL>
        <电话 类型="公司">(010)62345678</电话>
        <电话 类型="家庭">(010)88888888</电话>
        <地址>
            <街道>五街1234号</街道>
            <城市>北京市</城市>
            <省份>北京</省份>
        </地址>
    </联系人>

    <联系人>
        <姓名>李四</姓名>
        <ID>002</ID>
        <公司>B公司</公司>
        <EMAIL>[email protected]</EMAIL>
        <电话 类型="公司">(021)87654321</电话>
        <地址>
            <街道>南京路9876号</街道>
            <城市>上海</城市>
            <省份>上海</省份>
        </地址>
    </联系人>
</联系人列表>

2.为了找到张三的家庭的电话号码,使用下列代码段

package test;

import java.util.List;
import java.net.URL;
import org.jdom.*;
import org.jdom.xpath.*;
import org.jdom.input.*;
import java.io.*;
import java.util.*;


public class Test{
  public static void main(String[] args) throws Exception{

    SAXBuilder builder = new SAXBuilder();
    Document doc = builder.build("kk.xml");
    Element root = doc.getRootElement();

    XPath xpath = XPath.newInstance("//电话[../姓名='张三'][@类型='家庭']");
     List list = xpath.selectNodes(root);

     Iterator iter = list.iterator();
     while (iter.hasNext()) {
       Element item = (Element) iter.next();
       System.err.println(item.getText());
     }


  }

输出结果正是我们想要的:(010)88888888.怎么样,很方便吧:)。

8过呢,我这个表达式效率不一定就是最高的,你对xpath掌握得越好,查询效率越好。

希望这篇小小的文章对你有所启发,如有错误,敬请批评指正。



版权声明   给作者写信
本篇文章对您是否有帮助?  投票:         投票结果:     6        1


  评论人:yangwq    参与分: 6    专家分: 0 发表时间: 2003-12-19 下午2:48
怎样修改张三的家庭的电话号码??


我在网上找到用JDOM 9 beta包, 在JBuild 7中引用了jdom.jar、xerces.jar、jaxen-core.jar、jaxen-jdom.jar、saxpath.jar,可以XPath来快速定位xml节点。
但是没有修改节点值的例子?

我在doc 找到xpath类中 setVariable的方法。

下面是在 “找到张三的家庭的电话号码”基础上改为 “修改张三的家庭的电话号码,”使用下列代码段:

import java.util.List;
import java.net.URL;
import org.jdom.*;
import org.jdom.xpath.*;
import org.jdom.input.*;
import java.io.*;
import java.util.*;


public class Test{
  public static void main(String[] args) throws Exception{

    SAXBuilder builder = new SAXBuilder();
    Document doc = builder.build("cc.xml");
    Element root = doc.getRootElement();
    
    //找到张三的家庭的电话号码 
    XPath xpath = XPath.newInstance("//电话[../姓名='张三'][@类型='家庭']");

    List list = xpath.selectNodes(root);
    xpath.setVariable("家庭","12312313") ;
   
    /*
    Iterator iter = list.iterator();
    while (iter.hasNext()) {
      Element item = (Element) iter.next();
      System.err.println(item.getText());
    }
   */
  }
}

这段码无法”修改张三的家庭的电话号码 ",请教高手怎样实现修改?







  评论人:Jagie    参与分: 291    专家分: 1420    来自: 北京
发表时间: 2003-12-19 下午3:19
Xpath的setVariable方法是设置xpath表达式中的形参用的,不是用来改变节点的,你要改变节点,请使用Element的set方法。


Iterator iter = list.iterator();
while (iter.hasNext()) {
Element item = (Element) iter.next();
item.setText("aaaaaaaaaaaaaaaaaaa");
}


如果你想把对xml文档所做的改变保存下来,请参考一下代码!


XMLOutputter outputter = new XMLOutputter();
outputter.setTextTrim(true);
outputter.setIndent(" ");
outputter.setNewlines(true);
outputter.setEncoding("GB2312");

String s = outputter.output(doc,new FileOutputStream("aaa.xml"));


  评论人:djb_skyface    参与分: 36    专家分: 0 发表时间: 2004-6-4 上午10:44
    我试过楼主的方法,发现有个问题,是姓名前的".."导致的,我将..改为//就可以了,不知道是不是新版本的缘故(jdom10beta)。
下面是两种不同的途径:
1.我的途径
    SAXBuilder builder = new SAXBuilder();
    Document doc = builder.build("tel.xml");
    Element root = doc.getRootElement();
    XPath xpath = XPath.newInstance("//电话[//姓名='张三'][@类型='家庭']");
    Element e = (Element)xpath.selectSingleNode(root);
    //xpath.setVariable("//电话[//姓名='张三'][@类型='家庭']","33333333");
    System.out.println(e.getText());
2.楼主的途径
   SAXBuilder builder = new SAXBuilder();
    Document doc = builder.build("tel.xml");
    Element root = doc.getRootElement();
    XPath xpath = XPath.newInstance("//电话[//姓名='张三'][@类型='家庭']");
    List list = xpath.selectNodes(root);
    Iterator iter = list.iterator();
    while (iter.hasNext()) {
      Element item = (Element) iter.next();
      System.err.println(item.getText());
    }
两种途径都可以,但是我觉得我的途径理解起来较为自然,因为得到一个节点,不需要遍历的。

   另外,我试了楼主给出的Element.setText()方法,不能修改xml文件,不知为何,下面为源代码

    SAXBuilder builder = new SAXBuilder();
    Document doc = builder.build("tel.xml");
    Element root = doc.getRootElement();
    XPath xpath = XPath.newInstance("//电话[//姓名='张三'][@类型='家庭']");
    List list = xpath.selectNodes(root);
    Iterator iter = list.iterator();
    while (iter.hasNext()) {
      Element item = (Element) iter.next();
      item.setText("333333333");
      System.err.println(item.getText());
    }
  这段代码只能修改后面的打印结果,并不能修改xml文件里的值。

  评论人:djb_skyface    参与分: 36    专家分: 0 发表时间: 2004-6-4 上午10:55
请楼主原谅我的疏忽,没有看到楼主下面的话。
不过,楼主的代码似乎可以再简洁点:

楼主代码:

XMLOutputter outputter = new XMLOutputter();
outputter.setTextTrim(true);
outputter.setIndent(" ");
outputter.setNewlines(true);
outputter.setEncoding("GB2312");
String s = outputter.output(doc,new FileOutputStream("aaa.xml"));

我的代码:

      String indent = " ";//如果对上面的kk.xml文件修改,则不需再有indent
      boolean newLines = true;//不需再new line
      XMLOutputter outp = new XMLOutputter(indent, newLines, "GB2312");
      outp.setTextTrim(true);
      outp.output(doc, new FileOutputStream("aaa.xml"));

生成doc的

客人: eastasp 发表时间: 2004-6-4 上午11:38
 比如下面这个xml,我连Row中的Attribute属性的个数都得不到,请高手指点
<?xml version="1.0" standalone="yes" ?> 
- <!--  edited with XMLSPY v5 rel. 3 U (http://www.xmlspy.com) by heavyz (Nanjing University) 
  --> 
- <DATAPACKET Version="2.0">
- <METADATA>
- <FIELDS>
  <FIELD attrname="BsMt_MaterialID" fieldtype="string" width="20" /> 
  <FIELD attrname="BsMt_Barcode" fieldtype="string" width="30" /> 
  <FIELD attrname="BsMt_Name" fieldtype="string" width="50" /> 
  <FIELD attrname="BsMt_Specification" fieldtype="string" width="50" /> 
  <FIELD attrname="BsMt_Model" fieldtype="string" width="50" /> 
  <FIELD attrname="BsMt_Color" fieldtype="string" width="30" /> 
  <FIELD attrname="BsMt_UnitWeight" fieldtype="r8" width="22" /> 
  <FIELD attrname="BsMt_UnitName" fieldtype="string" width="5" /> 
  <FIELD attrname="BsMt_DepotID" fieldtype="string" width="50" /> 
  <FIELD attrname="BsMt_ProviderID" fieldtype="string" width="50" /> 
  <FIELD attrname="BsMt_Type" fieldtype="string" width="50" /> 
  <FIELD attrname="BsMt_Sort" fieldtype="string" width="50" /> 
  <FIELD attrname="BsMt_Note" fieldtype="string" width="200" /> 
  <FIELD attrname="BsMt_Reserve1" fieldtype="string" width="50" /> 
  <FIELD attrname="BsMt_Reserve2" fieldtype="string" width="50" /> 
  <FIELD attrname="BsMt_Reserve3" fieldtype="string" width="50" /> 
  <FIELD attrname="BsMt_Reserve4" fieldtype="string" width="50" /> 
  <FIELD attrname="BsMt_Reserve5" fieldtype="string" width="50" /> 
  </FIELDS>
  <PARAMS /> 
  </METADATA>
- <ROWDATA>
  <ROW BsMt_MaterialID="0002" BsMt_Barcode="0000000002" /> 
  <ROW BsMt_MaterialID="001" /> 
  <ROW BsMt_MaterialID="0015" BsMt_Barcode="0020" /> 
  <ROW BsMt_MaterialID="0016" BsMt_Barcode="0020" /> 
  <ROW BsMt_MaterialID=">" BsMt_Barcode="0020" /> 
  <ROW BsMt_MaterialID="0018" BsMt_Barcode="0020" /> 
  <ROW BsMt_MaterialID="0019" BsMt_Barcode="0020" /> 
  <ROW BsMt_MaterialID="002" /> 
  <ROW BsMt_MaterialID="0020" BsMt_Barcode="0020" /> 
  <ROW BsMt_MaterialID="002001" BsMt_Barcode="0000000" /> 
  <ROW BsMt_MaterialID="003" /> 
  <ROW BsMt_MaterialID="004" /> 
  <ROW BsMt_MaterialID="005" /> 
  <ROW /> 
  <ROW /> 
  </ROWDATA>
  </DATAPACKET>

你可能感兴趣的:(xpath)