Как распечатать заводной узел с сохраненным пространством имен? - PullRequest
4 голосов
/ 23 октября 2008

Когда я использую этот код для вывода некоторого XML, я проанализировал (и изменил) с XmlParser

XmlParser parser = new XmlParser()
def root = parser.parseText(feedUrl.toURL().text)
def writer = new StringWriter()
new XmlNodePrinter(new PrintWriter(writer)).print(root)
println writer.toString()

объявления пространства имен на корневом узле не печатаются, даже если они присутствуют в toString() из root ... какие-либо идеи?

Ответы [ 2 ]

3 голосов
/ 16 февраля 2011

У меня только что возникла та же проблема, и после небольшого перебора я нашел обходной путь.

Вы используете XmlSluper вместо XmlParser и используете StreamingMarkupBuilder вместо XmlNodePrinter . Затем вы используете замыкание в bind и используете встроенную переменную mkp для объявления пространств имен.

Например; используя исходный XML-пример Теда сверху:

def root = new XmlSlurper().parseText("http://stackoverflow.com/feeds/question/227447".toURL().text))
def outputBuilder = new StreamingMarkupBuilder()
String result = XmlUtil.serialize(outputBuilder.bind {
    mkp.declareNamespace('':'http://www.w3.org/2005/Atom')
    mkp.declareNamespace('creativeCommons':'http://backend.userland.com/creativeCommonsRssModule')
    mkp.declareNamespace('re':'http://purl.org/atompub/rank/1.0')
    mkp.yield root }
)
println result

Результаты:

<?xml version="1.0" encoding="UTF-8"?><feed xmlns:creativeCommons="http://backend.userland.com/creativeCommonsRssModule" xmlns="http://www.w3.org/2005/Atom" xmlns:re="http://purl.org/atompub/rank/1.0">
<title type="text">How do I print a groovy Node with namespace preserved? - Stack Overflow </title>
<link rel="self" type="application/atom+xml" href="http://stackoverflow.com/feeds/question/227447"/>
<link rel="alternate" type="text/html" href="http://stackoverflow.com/questions/227447"/>
<subtitle>most recent 30 from stackoverflow.com</subtitle>
<updated>2011-02-16T05:13:17Z</updated>
<id>http://stackoverflow.com/feeds/question/227447</id>
<creativeCommons:license>http://www.creativecommons.org/licenses/by-nc/2.5/rdf</creativeCommons:license>
<entry>
<id>/211328/kak-raspechatat-zavodnoi-uzel-s-sohranennym-prostranstvom-imen</id>
<re:rank scheme="http://stackoverflow.com">2</re:rank>
1 голос
/ 23 октября 2008

Похоже, что он денормализует вывод и включает в себя контекст пространства имен вместе с узлами, которым на самом деле нужен контекст пространства имен.

Например, веб-страница для этого вопроса имеет встроенное пространство имен creativeCommons:

<feed xmlns="http://www.w3.org/2005/Atom" xmlns:creativeCommons="http://backend.userland.com/creativeCommonsRssModule" xmlns:thr="http://purl.org/syndication/thread/1.0">
  <!-- snip -->
  <creativeCommons:license>http://www.creativecommons.org/licenses/by-nc/2.5/rdf</creativeCommons:license>
  <!-- snip -->
</feed>

Когда вы выводите xml, используя этот скрипт:

def root = new XmlParser().parseText("http://stackoverflow.com/feeds/question/227447".toURL().text)
println new XmlNodePrinter().print(root)

В итоге происходит перемещение пространства имен к узлу лицензии, которому необходимо это пространство имен. В этом случае не так уж и много, поскольку в этом пространстве имен только один узел. Если бы большая часть XML была пространством имен, это, вероятно, привело бы к еще большему скоплению вещей.

<feed xmlns="http://www.w3.org/2005/Atom">
  <!-- snip -->
    <creativeCommons:license xmlns:creativeCommons="http://backend.userland.com/creativeCommonsRssModule">
http://www.creativecommons.org/licenses/by-nc/2.5/rdf
  </creativeCommons:license>
  <!-- snip -->
</feed>

Если вы действительно хотите, чтобы узлы были нормализованы, вам нужно было бы внести некоторые изменения в XmlNodePrinter, чтобы сделать 2 прохода через XML, сначала собрать все используемые пространства имен, а 2-й - вывести их сверху, а не внутри Пространство имен Отличный исходный код на самом деле довольно читабелен и его не так сложно изменить, если вам это действительно нужно.

...