在任何類似的問題中,可以使用Kayessian公式爲nodeset交集。
如果我們有兩個節點集$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>
這是偉大的,完美的作品! – ScottD