5

このコードを使用して、解析(および変更)した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()ルートノードの名前空間宣言は、ルートのにあるにもかかわらず、出力されません...何かアイデアはありますか?

4

2 に答える 2

4

私はちょうど同じ問題を抱えていましたが、少しいじった後、回避策を見つけました。

XmlParserの代わりにXmlSluperを使用し、XmlNodePrinterの代わりにStreamingMarkupBuilderを使用します。次に、bindのクロージャーを利用し、mkp組み込み変数を使用して名前空間を宣言します。

例えば; 上記の Ted のソース 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>http://stackoverflow.com/questions/227447/how-do-i-print-a-groovy-node-with-namespace-preserved</id>
<re:rank scheme="http://stackoverflow.com">2</re:rank>
于 2011-02-16T05:25:26.847 に答える
1

出力を非正規化し、実際に名前空間コンテキストを必要とするノードとともに名前空間コンテキストを含めているようです。

たとえば、この質問の Web ページには、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 を微調整して、XML を介して 2 つのパスを実行する必要があります。最初に、使用されているすべての名前空間を収集し、2 番目に、各名前空間内ではなく上部にそれらを出力します。グルーヴィーなソース コードは実際には非常に読みやすく、実際にこれが必要な場合でも変更するのはそれほど難しくありません。

于 2008-10-23T04:55:15.527 に答える