gpt4 book ai didi

xslt - 处理指令变换

转载 作者:行者123 更新时间:2023-12-02 11:42:00 25 4
gpt4 key购买 nike

我想将源 xml 中的处理指令转换为输出中的某个标记

输入

<?xml version="1.0" encoding="utf-8"?>
<root>
<?PI_start?> SOME TEXT <?PI_end?>
</root>

我想要这样的输出 xml

<root>
<tag> SOME TEXT </tag>
</root>

我可以做吗?如果是,我必须使用什么 xsl 进行转换?

我只找到了一种将 PI 转换为开始和结束标记的方法。 PI可以包含一些内容。

输入 XML

<root>
<?PI SOME TEXT?>
</root>

XSL

<xsl:template match="processing-instruction('PI')">
<tag><xsl:value-of select="."/></tag>
</xsl:template>

输出

<tag>SOME TEXT</tag>

但这有点不是我的情况

最佳答案

这种转变:

<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output omit-xml-declaration="yes" indent="yes"/>
<xsl:strip-space elements="*"/>

<xsl:template match="node()|@*">
<xsl:copy>
<xsl:apply-templates select="node()|@*"/>
</xsl:copy>
</xsl:template>

<xsl:template match="processing-instruction('PI_start')">
<tag>
<xsl:apply-templates mode="copy" select=
"following-sibling::node()[1][self::text()]"/>
</tag>
</xsl:template>

<xsl:template match=
"processing-instruction('PI_end')
|
text()[preceding-sibling::node()[1]
[self::processing-instruction('PI_start')]]
"/>
</xsl:stylesheet>

应用于提供的 XML 文档时:

<?xml version="1.0" encoding="utf-8"?>
<root>
<?PI_start?> SOME TEXT <?PI_end?>
</root>

产生想要的正确结果:

<root>
<tag> SOME TEXT </tag>
</root>

请注意:

  1. 身份规则用于“按原样”复制所有节点。

  2. 我们仅针对应以某种方式更改的节点提供了额外的模板

  3. 与第一个 PI 匹配的模板“几乎完成了所有工作”。它创建了 tag元素,并将模板应用于以下同级节点(如果它是 PI)。

  4. 我们以“复制”模式为第一个 PI 的直接兄弟文本节点应用模板

  5. “复制”模式未在任何地方声明,这会导致选择处理文本节点的默认模板 - 它的操作只是复制文本节点。这是一个技巧,使我们无需在“复制”模式下定义模板。

  6. 我们有一个空模板,实际上会删除不需要的节点:第二个 PI 以及第一个 PI 的直接兄弟文本节点的第二个副本。

更新:OP 表示他也对两个 PI 之间可能存在不同节点(不仅仅是文本节点)的情况感兴趣。

这是一项更加复杂的任务,这里有一个解决方案:

<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output omit-xml-declaration="yes" indent="yes"/>
<xsl:strip-space elements="*"/>

<xsl:key name="kSurrounded" match="node()"
use="concat(
generate-id(preceding-sibling::processing-instruction('PI_start')[1]),
'+++',
generate-id(following-sibling::processing-instruction('PI_end')[1])
)"/>

<xsl:template match="node()|@*" name="identity">
<xsl:copy>
<xsl:apply-templates select="node()|@*"/>
</xsl:copy>
</xsl:template>

<xsl:template match="processing-instruction('PI_start')">
<tag>
<xsl:apply-templates mode="copy" select=
"key('kSurrounded',
concat(generate-id(),
'+++',
generate-id(following-sibling::processing-instruction('PI_end')[1])
)
)"/>
</tag>
</xsl:template>

<xsl:template match=
"processing-instruction('PI_end')
|
node()[(preceding-sibling::processing-instruction('PI_start')
|
preceding-sibling::processing-instruction('PI_end')
)
[last()][self::processing-instruction('PI_start')]
and
(following-sibling::processing-instruction('PI_start')
|
following-sibling::processing-instruction('PI_end')
)
[1][self::processing-instruction('PI_end')]
]
"/>

<xsl:template match="node()" mode="copy">
<xsl:call-template name="identity"/>
</xsl:template>
</xsl:stylesheet>

当上述转换应用于以下 XML 文档时:

<root>
<?PI_start?> <strong>Some</strong> TEXT <?PI_end?> XA <?PI_end?>
</root>

产生了想要的正确输出:

<root>
<tag>
<strong>Some</strong> TEXT
</tag> XA
</root>

关于xslt - 处理指令变换,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/4165325/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com