我正在使用boost序列化来实现持久性,并且由于库不支持保存到旧版本的存档/数据结构的想法,我虽然我会给XSLT和XPath一个新的转换根据需要将版本更新为旧版本 .
我已经走了一半左右,但似乎无法完成它(这也是我第一次尝试XSLT和XPath / XQuery,所以请原谅任何明显的错误) .
这是我的首发XML:
<?xml version="1.0" encoding="UTF-8" standalone="yes" ?>
<!DOCTYPE boost_serialization>
<boost_serialization signature="serialization::archive" version="7">
<tester class_id="0" tracking_level="0" version="0">
<count>2</count>
<item_version>0</item_version>
<item class_id="2" class_name="CLASS_D" tracking_level="0" version="0">
<A class_id="1" tracking_level="1" version="0" object_id="_0">
<pimpl class_id="3" tracking_level="1" version="0" object_id="_1">
<b>1</b>
</pimpl>
</A>
<pimpl class_id="4" tracking_level="1" version="0" object_id="_2">
<c>2</c>
</pimpl>
</item>
<item class_id="5" class_name="CLASS_E" tracking_level="0" version="0">
<A object_id="_3">
<pimpl class_id_reference="3" object_id="_4">
<b>1</b>
</pimpl>
</A>
<pimpl class_id="6" tracking_level="1" version="0" object_id="_5">
<f>2</f>
</pimpl>
</item>
</tester>
</boost_serialization>
我想要做的是,使用属性class_name =“CLASS_E”将项目转换为类似于class_name =“CLASS_D”的项目,但我需要单独保留object_id属性 .
这就是我要的:
<?xml version="1.0" encoding="utf-8"?>
<boost_serialization signature="serialization::archive" version="7">
<tester class_id="0" tracking_level="0" version="0">
<count>2</count>
<item_version>0</item_version>
<item class_id="2" class_name="CLASS_D" tracking_level="0" version="0">
<A class_id="1" tracking_level="1" version="0" object_id="_0">
<pimpl class_id="3" tracking_level="1" version="0" object_id="_1">
<b>1</b>
</pimpl>
</A>
<pimpl class_id="4" tracking_level="1" version="0" object_id="_2">
<c>2</c>
</pimpl>
</item>
<item class_name="CLASS_D" class_id="2" tracking_level="0" version="0">
<A object_id="_3">
<pimpl class_id_reference="3" object_id="_4">
<b>1</b>
</pimpl>
</A>
<pimpl class_id="4" tracking_level="1" version="0" object_id="_5">
<c>2</c>
</pimpl>
</item>
</tester>
</boost_serialization>
这是我到目前为止的模板:
<?xml version="1.0" encoding="ISO-8859-1"?>
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="xml" version="1.0" omit-xml-declaration="no" encoding="UTF-8" indent="yes"/>
<!-- identity-->
<xsl:template match="@*|node()">
<xsl:copy>
<xsl:apply-templates select="@*|node()"/>
</xsl:copy>
</xsl:template>
<!-- replace attribute class_name value with another-->
<!-- replace attribute class_id value with another-->
<!-- only on this node!-->
<!-- could call another template to change more nested things-->
<xsl:template match="item/@class_name[. = 'CLASS_E']">
<xsl:attribute name="class_name">CLASS_D</xsl:attribute>
<xsl:attribute name="class_id">2</xsl:attribute>
</xsl:template>
</xsl:stylesheet>
我不确定如何继续编辑我与该行匹配的项的子节点:因为我需要将“f”节点更改为“c”并将pimpl“class_id”从6更改为4
提前致谢
1 回答
您刚刚匹配的项目是一个属性,因此没有子节点!父节点将与标识模板匹配,因此要进行'continue'编辑,只需要一个与您希望匹配的 class_id 属性匹配的模板,但在xpath表达式中包含相关 item 的匹配项 .
例如,要更改 pimpl 元素的 class_id ,请添加此模板
要更改 f 元素,请添加此模板
请记住,匹配适用于输入文档,因此在输出文档中将“CLASS_E”更改为“CLASS_D”无关紧要 .
请注意,您当前模板可能存在与 class_name 属性匹配的问题 . 在此,您将使用两个属性替换它,包括已存在的 class_id 属性 . XSLT将替换输出树中的属性,其中已经输出了具有相同名称的属性 . 这意味着如果您的XML实际上看起来像这样......
然后它将输出实际看起来这个
这是因为在匹配 class_name 的模板之后,标识模板将匹配 class_id 属性,因此将替换当前与之匹配的 class_id 属性 .
但是长话短说,试试这个XSLT .....