вставить / заменить XML-тег в XmlDocument

голоса
3

У меня есть XmlDocumentв Java, созданный с Weblogic XmlDocumentанализатором.

Я хочу , чтобы заменить содержимое тега в этом XMLDocumentсо своими собственными данными, или вставить тег , если он не существует.

<customdata>
   <tag1 />
   <tag2>mfkdslmlfkm</tag2>
   <location />
   <tag3 />
</customdata>

Я, например, хочу, чтобы вставить некоторые URL в месте тега:

<location>http://something</location>

но в противном случае оставить XML как есть.

В настоящее время я использую XMLCursor:

    XmlObject xmlobj = XmlObject.Factory.parse(a.getCustomData(), options);
    XmlCursor xmlcur = xmlobj.newCursor();

    while (xmlcur.hasNextToken()) {
      boolean found = false;
      if (xmlcur.isStart() && schema-location.equals(xmlcur.getName().toString())) {
        xmlcur.setTextValue(http://replaced);
        System.out.println(replaced);
        found = true;
      } else if (xmlcur.isStart() && customdata.equals(xmlcur.getName().toString())) {
        xmlcur.push();
      } else if (xmlcur.isEnddoc()) {
        if (!found) {
          xmlcur.pop();
          xmlcur.toEndToken();
          xmlcur.insertElementWithText(schema-location, http://inserted);
          System.out.println(inserted);
        }

      }
      xmlcur.toNextToken();
    }

Я пытался найти «быстрый» xqueryспособ сделать это , так как XmlDocumentесть execQueryметод, но не нашел его очень легко.

Есть ли у кого-нибудь есть лучший способ, чем это? Это кажется немного сложным.

Задан 19/08/2008 в 09:46
источник пользователем
На других языках...                            


4 ответов

голоса
4

Как насчет подхода, основанного на XPath? Мне нравится этот подход, поскольку логика супер-легко понять. Код довольно много самодокументирован.

Если XML-документ доступен для вас в качестве объекта org.w3c.dom.Document (как вернуть большинство парсеры), то вы могли бы сделать что-то вроде следующего:

// get the list of customdata nodes
NodeList customDataNodeSet = findNodes(document, "//customdata" );

for (int i=0 ; i < customDataNodeSet.getLength() ; i++) {
  Node customDataNode = customDataNodeSet.item( i );

  // get the location nodes (if any) within this one customdata node
  NodeList locationNodeSet = findNodes(customDataNode, "location" );

  if (locationNodeSet.getLength() > 0) {
    // replace
    locationNodeSet.item( 0 ).setTextContent( "http://stackoverflow.com/" );
  }
  else {
    // insert
    Element newLocationNode = document.createElement( "location" );
    newLocationNode.setTextContent("http://stackoverflow.com/" );
    customDataNode.appendChild( newLocationNode );
  }
}

И вот вспомогательный метод findNodes, что делает поиск XPath.

private NodeList findNodes( Object obj, String xPathString )
  throws XPathExpressionException {

  XPath xPath = XPathFactory.newInstance().newXPath();
  XPathExpression expression = xPath.compile( xPathString );
  return (NodeList) expression.evaluate( obj, XPathConstants.NODESET );
}
Ответил 19/08/2008 в 12:31
источник пользователем

голоса
1

Если вы не знаете схему решения XStream, вероятно, не путь. По крайней мере, XStream на радаре теперь может пригодиться в будущем!

Ответил 19/08/2008 в 16:15
источник пользователем

голоса
1

Как насчет подхода объектно-ориентированного? Вы можете deserialise в XML для объекта, установите значение местоположения на объекте, а затем сериализация обратно в XML.

XStream делает это очень легко.

Например, нужно определить основной объект, который в вашем случае является CustomData (я использую открытые поля, чтобы не усложнять пример простой):

public class CustomData {
  public String tag1;
  public String tag2;
  public String location;
  public String tag3;
}

Тогда вы инициализации XStream:

XStream xstream = new XStream();
// if you need to output the main tag in lowercase, use the following line
xstream.alias("customdata", CustomData.class);  

Теперь вы можете построить объект из XML, в поле местоположения на объекте и регенерировать XML:

CustomData d = (CustomData)xstream.fromXML(xml);
d.location = "http://stackoverflow.com";
xml = xstream.toXML(d);

Как это звучит?

Ответил 19/08/2008 в 11:45
источник пользователем

голоса
0

Вы должны быть в состоянии сделать это с query

пытаться

 fn:replace(string,pattern,replace)

Я новичок в XQUERY себя, и я нашел, что это будет болезненный язык запросов для работы, но это действительно работает тихо хорошо, как только вы получите в течение начальной кривой обучения.

Я все еще хотел было простой способ, который был столь же эффективным?

Ответил 19/08/2008 в 11:38
источник пользователем

Cookies help us deliver our services. By using our services, you agree to our use of cookies. Learn more