Сведения о вопросе

Ислам

21:08, 8th August, 2020

Теги

java   xml   xquery    

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

Просмотров: 453   Ответов: 4

У меня есть XmlDocument в java, созданный с помощью парсера Weblogic XmlDocument .

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

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

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

<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();
    }

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

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



  Сведения об ответе

lourence

12:45, 7th August, 2020

Как насчет подхода, основанного на 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 );
}


  Сведения об ответе

fo_I_K

16:22, 18th August, 2020

Как насчет объектно-ориентированного подхода? Вы можете десериализовать 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, установить поле location для объекта и повторно создать XML:

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

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


  Сведения об ответе

ITSME

14:15, 9th August, 2020

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


  Сведения об ответе

PHPH

22:03, 20th August, 2020

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

пробовать

 fn:replace(string,pattern,replace)

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

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


Ответить на вопрос

Чтобы ответить на вопрос вам нужно войти в систему или зарегистрироваться