2009-06-10 3 views
2

Я написал класс для разбора некоторого xml в объект, и он работает некорректно, когда я пытаюсь получить значение узла, я получаю нуль, а не содержимое узла.Почему этот Java-код не правильно разбирает мой Xml?

Вот упрощенная версия моего класса, который просто делает XML разбора одного узла:

import java.io.File; 
import java.io.IOException; 
import java.util.ArrayList; 

import javax.xml.parsers.DocumentBuilder; 
import javax.xml.parsers.DocumentBuilderFactory; 
import javax.xml.parsers.ParserConfigurationException; 

import org.apache.log4j.Logger; 
import org.apache.log4j.xml.DOMConfigurator; 
import org.w3c.dom.Document; 
import org.w3c.dom.Element; 
import org.w3c.dom.Node; 
import org.w3c.dom.NodeList; 
import org.xml.sax.SAXException; 

public class XmlReaderCutDown { 

    private static Logger logger = Logger.getLogger(CtiButtonsXmlReader.class); 

    public static void testXml(String confFile){ 
     DocumentBuilderFactory docBuilderFactory = DocumentBuilderFactory.newInstance(); 
     try { 
      DocumentBuilder docBuilder = docBuilderFactory.newDocumentBuilder(); 
      Document doc = docBuilder.parse(new File(confFile)); 
      doc.getDocumentElement().normalize(); 
      if (logger.isDebugEnabled()){ 
       logger.debug("The root element is " + doc.getDocumentElement().getNodeName()); 
      } 

      NodeList rows = doc.getElementsByTagName("row"); 

      NodeList topRowButtons = ((Element)rows.item(0)).getElementsByTagName("button"); 
      logger.debug("Top row has " +topRowButtons.getLength() + " items."); 
      Node buttonNode = topRowButtons.item(0); 

      NodeList nodeList = ((Element)buttonNode).getElementsByTagName("id"); 
      logger.debug("Node list count for "+ "id" + " = " + nodeList.getLength()); 
      Element element = (Element)nodeList.item(0); 
      String xx = element.getNodeValue(); 
      logger.debug(xx); 
      String elementValue = ((Node)element).getNodeValue(); 
      if (elementValue != null) { 
       elementValue = elementValue.trim(); 
      } 
      else { 
       logger.debug("Value was null"); 
      } 
      logger.debug("Node id = "+ elementValue); 

     } catch (ParserConfigurationException e) { 

      logger.fatal(e.toString(),e); 
     } catch (SAXException e) { 
      logger.fatal(e.toString(),e); 
     } catch (IOException e) { 
      logger.fatal(e.toString(),e); 
     } catch (Exception e) { 
      logger.fatal(e.toString(),e); 
     } 

    } 


    public static void main(String[] args){ 
     DOMConfigurator.configure("log4j.xml"); 

     testXml("test.xml"); 
    } 
} 

А вот урезанная версия моего файла XML:

<?xml version="1.0"?> 
<root> 
    <row> 
     <button> 
      <id>this is an id</id> 
      <action>action</action> 
      <image-src>../images/img.png</image-src> 
      <alt-text>alt txt</alt-text> 
      <tool-tip>Tool tip</tool-tip> 
     </button> 
    </row> 
</root> 

Это является то, что выход лесозаготовительных statments:

DEBUG XmlReaderCutDown - корневой элемент Root

DEBUG XmlReaderCutDown - Верхняя строка содержит 1 элемент.

DEBUG XmlReaderCutDown - счетчик список узлов для ид = 1

DEBUG XmlReaderCutDown -

DEBUG XmlReaderCutDown - значение было нулевым

DEBUG XmlReaderCutDown - Node ID = нулевой

Почему не он получает текст в узле xml?

Я бегу с помощью JDK 1.6_10

+0

Примечания, если вы хотите попробовать его, но не имеют log4j, то вы можете заменить лесозаготовительный statments с системой .out.println(); –

ответ

6

Поскольку элемент вы получаете родительский элемент текста один, содержащий текст, который нужно. Чтобы получить доступ к тексту этого элемента, вы должны использовать getTextContent вместо getNodeValue.

Для получения дополнительной информации см. Таблицу в файле Node javadoc.

+0

Спасибо, что работает шарм, один вопрос, хотя почему код на этой странице http://www.java-tips.org/java-se-tips/javax.xml.parsers/how-to-read-xml-file -in-java.html работать, когда он использует значение узла? –

+0

Потому что они получают первый дочерний элемент рассматриваемого элемента, который является текстовым узлом. Это глобально та же самая вещь, что и getTextContent, только безопаснее в случае getTextContent. –

+0

Вы можете использовать nodeValue() в текстовом узле. –

1

Рассмотрите возможность использования XPath в качестве альтернативы ручной ходьбы узлов:

public static void testXml(String confFile) 
     throws XPathExpressionException { 
    XPathFactory xpFactory = XPathFactory.newInstance(); 
    XPath xpath = xpFactory.newXPath(); 

    NodeList rows = (NodeList) xpath.evaluate("root/row", 
     new InputSource(confFile), XPathConstants.NODESET); 

    for (int i = 0; i < rows.getLength(); i++) { 
     Node row = rows.item(i); 
     String id = xpath.evaluate("button/id", row); 
     System.out.println("id=" + id); 
    } 
    } 

    public static void main(String[] args) 
     throws XPathExpressionException { 
    testXml("test.xml"); 
    } 
Смежные вопросы