在任何类似的问题中,都可以使用 Kayessian 公式计算节点集交集。
如果我们有两个节点集$ns1 和$ns2,那么属于这两个节点集的所有节点都被这个 XPath 表达式选择:
$ns1[count(.|$ns2) = count($ns2)]
在您的情况下,您只需将 $ns1 替换为:
//w:r[w:fldChar/@w:fldCharType='begin'][1]/following-sibling::*
..
并将$ns2 替换为:
//w:r[w:fldChar/@w:fldCharType='end'][1]/preceding-sibling::*
生成的 XPath 表达式可能看起来过于复杂,但您获得的是能够非常轻松且几乎机械地解决任何此类问题的能力:
/*/w:r
[w:fldChar/@w:fldCharType='begin']/following-sibling::*
[count(. | /*/w:r[w:fldChar/@w:fldCharType='end']
/preceding-sibling::*
)
=
count(/*/w:r[w:fldChar/@w:fldCharType='end']
/preceding-sibling::*)
]
基于 XSLT 的验证:
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:w="some:namespace">
<xsl:output omit-xml-declaration="yes" indent="yes"/>
<xsl:strip-space elements="*"/>
<xsl:template match="/">
<xsl:copy-of select=
"/*/w:r
[w:fldChar/@w:fldCharType='begin']/following-sibling::*
[count(. | /*/w:r[w:fldChar/@w:fldCharType='end']
/preceding-sibling::*
)
=
count(/*/w:r[w:fldChar/@w:fldCharType='end']
/preceding-sibling::*)
]
"/>
</xsl:template>
</xsl:stylesheet>
当此转换应用于此 XML 文档时:
<t xmlns:w="some:namespace">
<w:r>
<w:fldChar w:fldCharType="before-begin"/>
</w:r>
<w:r>
<w:fldChar w:fldCharType="begin"/>
</w:r>
<w:r>
<w:instrText> DOCPROPERTY EvidenceBase \* MERGEFORMAT </w:instrText>
</w:r>
<w:r>
<w:fldChar w:fldCharType="separate"/>
</w:r>
<w:r>
<w:t>EvidenceBase</w:t>
</w:r>
<w:r>
<w:fldChar w:fldCharType="end"/>
</w:r>
<w:r>
<w:fldChar w:fldCharType="after-end"/>
</w:r>
</t>
正是需要的元素被选中并复制到输出中:
<w:r xmlns:w="some:namespace">
<w:instrText> DOCPROPERTY EvidenceBase \* MERGEFORMAT </w:instrText>
</w:r>
<w:r xmlns:w="some:namespace">
<w:fldChar w:fldCharType="separate"/>
</w:r>
<w:r xmlns:w="some:namespace">
<w:t>EvidenceBase</w:t>
</w:r>