0

私はチュートリアルを進めていて、うまくいくと思っていた問題に遭遇しました。誰かがこのコードを手伝ってくれることを願っています。私はこれを乗り越えることができません。以下で共有するこのコードは、XPATH式 " // * / text() "を使用すると一致しますが、それよりも具体的にして " // tag0:G / text() "を使用すると一致しません。私が間違っていることについて何か考えはありますか?以下に提供されているXMLから2つの「tag0:G」値を取得しようとしています。

import java.io.IOException;
import java.io.StringReader;    
import javax.xml.parsers.DocumentBuilderFactory;
import javax.xml.parsers.ParserConfigurationException;
import javax.xml.xpath.XPathConstants;
import javax.xml.xpath.XPathExpression;
import javax.xml.xpath.XPathExpressionException;
import javax.xml.xpath.XPathFactory;    
import org.w3c.dom.Document;
import org.w3c.dom.NodeList;
import org.xml.sax.InputSource;
import org.xml.sax.SAXException;    
public class Test1 {        
  public static void main(String[] args) {
    System.out.println("Test start...");
    String myXML = "<?xml version=\"1.0\" encoding=\"UTF-8\"?>" +
    "<soapenv:Envelope xmlns:soapenv=\"http://schemas.xmlsoap.org/soap/envelope/\">" +
    "<soapenv:Body>" +
    "<tag0:getA xmlns:tag0=\"http://me.ws.ix\">" +
    "<tag0:B>" +
    "<tag0:CC>" +
    "<tag0:CC>" +
    "<tag0:D>false</tag0:D>" +
    "<tag0:E>false</tag0:E>" +
    "<tag0:F xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\" xsi:nil=\"true\"/>" +
    "<tag0:G>10001</tag0:G>" +
    "<tag0:H>7744000002</tag0:H>" +
    "</tag0:CC>" +
    "<tag0:CC>" +
    "<tag0:D>false</tag0:D>" +
    "<tag0:E>false</tag0:E>" +
    "<tag0:F xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\" xsi:nil=\"true\"/>" +
    "<tag0:G>20002</tag0:G>" +
    "<tag0:H>1111122222</tag0:H>" +
    "</tag0:CC>" +
    "</tag0:CC>" +
    "<tag0:I>2012-05-27 23:38:48</tag0:I>" +
    "</tag0:B>" +
    "</tag0:getA>" +
    "</soapenv:Body>" +
    "</soapenv:Envelope>";

    DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();
    factory.setNamespaceAware(true);
    Document doc = null;
    NodeList nodes = null;
    try {           
      doc = factory.newDocumentBuilder().parse( new InputSource( new StringReader( myXML) ) );
      XPathExpression expr = XPathFactory.newInstance().newXPath()
        .compile("//tag0:G/text()"); // this fails, I don't know why
      nodes = (NodeList)expr.evaluate(doc, XPathConstants.NODESET);
    } catch (SAXException e) {
      e.printStackTrace();
    } catch (IOException e) {
      e.printStackTrace();
    } catch (ParserConfigurationException e) {
      e.printStackTrace();
    } catch (XPathExpressionException e) {
      e.printStackTrace();
    }
    System.out.println("Nodes length: " + nodes.getLength() );
    for (int i = 0; i < nodes.getLength(); i++) { 
      String val = nodes.item(i).getNodeValue(); 
      System.out.println( "Val: " + val ); 
    }
    System.out.println("Test end...");
  }    
}
4

2 に答える 2

2

XPath式のコンパイルに取り掛かる前NamespaceContextに、XPathインスタンスにを登録する必要があります。XPath#setNamespaceContext(NamespaceContex nsContextを使用して行います 。

xml.apache.orgでNamespaceContextの章を使用して読んでください。

ドキュメントが名前空間情報で解析されている間、XPathエグゼキュータは、使用しているプレフィックスを、それを実行するドキュメントの名前空間URIと一致させる方法を知る必要があるという考え方です。プレフィックス自体は何も言いません。同じURIにマップされている限り、任意のプレフィックスを使用できます。

psは、XMLの名前空間に関する詳細です。

于 2012-05-28T00:37:32.090 に答える
0

もちろん、ごまかすこともできます-「G」(またはあなたが後にしている要素)のすべての出現が同じ名前空間からのものである場合-あなたは簡単に行うことができます

//*[local-name()='G'] 

ただし、これは機能しますが、これは問題を回避しているだけです。弾丸をかじって名前名を越えた方がいいかもしれません...

于 2012-05-28T04:11:00.880 に答える