XSLT 使用新值复制节点和所有子节点和属性
XSLT Duplicating node & all child nodes & attributes with new values
美好的一天....
我正在尝试使用 updated/new 元素文本 and/or 属性值复制节点。
我输入的XML文件:
<?xml version="1.0"?>
<products author="Jesper">
<product id="p1">
<name>Delta</name>
<price>800</price>
<stock>4</stock>
<country>Denmark</country>
</product>
</products>
所需的XML输出:
<?xml version="1.0" encoding="utf-8"?>
<products author="Jesper">
<product id="p1">
<name>Delta</name>
<price>800</price>
<stock>4</stock>
<country>Denmark</country>
</product>
<product id="NEW_p1">
<name>NEW_Delta</name>
<price>NEW_800</price>
<stock>NEW_4</stock>
<country>NEW_Denmark</country>
</product>
</products>
一段时间后,我目前拥有的XSLT如下:
<?xml version="1.0" encoding="utf-8"?>
<xsl:stylesheet version="2.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:msxsl="urn:schemas-microsoft-com:xslt"
exclude-result-prefixes="msxsl">
<xsl:output method="xml" indent="yes"/>
<xsl:template match="@*|node()">
<xsl:copy>
<xsl:apply-templates select="@*|node()"/>
</xsl:copy>
</xsl:template>
<xsl:template match ="product">
<xsl:copy>
<xsl:apply-templates select="@*|node()"/>
</xsl:copy>
<product>
<xsl:attribute name ="id">
<xsl:value-of select ="concat('NEW_',@id"/>
</xsl:attribute>
<xsl:copy>
<xsl:apply-templates select="node()"/>
</xsl:copy>
</product>
</xsl:template>
但是,使用上面的转换,我得到以下 XML 输出:
<?xml version="1.0" encoding="utf-8"?>
<products author="Jesper">
<product id="p1">
<name>Delta</name>
<price>800</price>
<stock>4</stock>
<country>Denmark</country>
</product>
<product id="NEW_p1"><product>
<name>Delta</name>
<price>800</price>
<stock>4</stock>
<country>Denmark</country>
</product></product>
</products>
如您所见,添加了 product 元素,同时我声明了一个具有新 @id 值的新 product 元素。因为我用来处理子节点,所以我相信这会再次处理 product 元素。
此外,我需要帮助更新子节点的值(在每个值前添加 'NEW_')。在这个网站上搜索大量问题,我相信我需要一个这样的模板:
<xsl:template match="*">
<xsl:element name ="{local-name()}">
<!--for all attributes-->
<xsl:copy-of select ="@*"/>
<xsl:value-of select = "."/>
</xsl:element>
</xsl:template>
提前感谢您 suggestions/ideas 我的问题。
已更新
感谢@Mathias 对我最初问题的回答。提供的答案又带来了一个涉及递归到 XML 结构的更深层次的问题。
输入XML文件:
<products author="Jesper">
<product id="p1">
<name>Delta
<innerName>MiddleDelta
<baseName>FinalDelta</baseName>
</innerName>
</name>
<price>800</price>
<stock>4</stock>
<country>Denmark
<city>Copenhagen</city>
</country>
</product>
</products>
更新后的期望输出文件如下:
<?xml version="1.0" encoding="utf-8"?>
<products author="Jesper">
<product id="p1">
<name>Delta
<innerName>MiddleDelta
<baseName>FinalDelta</baseName>
</innerName>
</name>
<price>800</price>
<stock>4</stock>
<country>Denmark
<city>Copenhagen</city>
</country>
</product>
<product id="NEW_p1">
<name>NEW_Delta
<innerName>NEW_MiddleDelta
<baseName>NEW_FinalDelta</baseName>
</innerName>
</name>
<price>NEW_800</price>
<stock>NEW_4</stock>
<country>NEW_Denmark
<city>NEW_Copenhagen</city>
</country>
</product>
</products>
我只能猜测使用模板会起作用,因为看到每个节点都有不同级别的子节点。在此先感谢您ideas/suggestions。
你已经不远了,但主要有两个问题:
- 您在第二个文字
product
元素中使用 xsl:copy
,这会在输出 中产生一个额外的 product
元素
- 您需要一种方法来找到
product
元素的所有子元素,再次输出它们并将 "NEW_" 添加到它们的文本内容中。
您确定 version
属性应设置为“2.0”吗?另外,我不确定这个练习的要点是什么...
XSLT 样式表
<?xml version="1.0" encoding="utf-8"?>
<xsl:stylesheet version="2.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="xml" indent="yes"/>
<xsl:template match="@*|node()">
<xsl:copy>
<xsl:apply-templates select="@*|node()"/>
</xsl:copy>
</xsl:template>
<xsl:template match ="product">
<xsl:copy>
<xsl:apply-templates select="@*|node()"/>
</xsl:copy>
<product id="{concat('NEW_',@id)}">
<xsl:for-each select="*">
<xsl:copy>
<xsl:value-of select="concat('NEW_',.)"/>
</xsl:copy>
</xsl:for-each>
</product>
</xsl:template>
</xsl:stylesheet>
XML输出
<?xml version="1.0" encoding="UTF-8"?>
<products author="Jesper">
<product id="p1">
<name>Delta</name>
<price>800</price>
<stock>4</stock>
<country>Denmark</country>
</product>
<product id="NEW_p1">
<name>NEW_Delta</name>
<price>NEW_800</price>
<stock>NEW_4</stock>
<country>NEW_Denmark</country>
</product>
</products>
这是对您更新后的问题的回应(恕我直言,应该将其作为一个新问题提出):
XSLT 1.0
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>
<xsl:strip-space elements="*"/>
<xsl:template match="/products">
<xsl:copy>
<xsl:copy-of select="product"/>
<xsl:apply-templates select="product"/>
</xsl:copy>
</xsl:template>
<!-- modified identity transform -->
<xsl:template match="*">
<xsl:copy>
<xsl:apply-templates select="@*|node()"/>
</xsl:copy>
</xsl:template>
<xsl:template match="@*">
<xsl:attribute name="{name()}">
<xsl:value-of select="concat('NEW_', .)"/>
</xsl:attribute>
</xsl:template>
<xsl:template match="text()">
<xsl:value-of select="concat('NEW_', .)"/>
</xsl:template>
</xsl:stylesheet>
美好的一天.... 我正在尝试使用 updated/new 元素文本 and/or 属性值复制节点。
我输入的XML文件:
<?xml version="1.0"?>
<products author="Jesper">
<product id="p1">
<name>Delta</name>
<price>800</price>
<stock>4</stock>
<country>Denmark</country>
</product>
</products>
所需的XML输出:
<?xml version="1.0" encoding="utf-8"?>
<products author="Jesper">
<product id="p1">
<name>Delta</name>
<price>800</price>
<stock>4</stock>
<country>Denmark</country>
</product>
<product id="NEW_p1">
<name>NEW_Delta</name>
<price>NEW_800</price>
<stock>NEW_4</stock>
<country>NEW_Denmark</country>
</product>
</products>
一段时间后,我目前拥有的XSLT如下:
<?xml version="1.0" encoding="utf-8"?>
<xsl:stylesheet version="2.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:msxsl="urn:schemas-microsoft-com:xslt"
exclude-result-prefixes="msxsl">
<xsl:output method="xml" indent="yes"/>
<xsl:template match="@*|node()">
<xsl:copy>
<xsl:apply-templates select="@*|node()"/>
</xsl:copy>
</xsl:template>
<xsl:template match ="product">
<xsl:copy>
<xsl:apply-templates select="@*|node()"/>
</xsl:copy>
<product>
<xsl:attribute name ="id">
<xsl:value-of select ="concat('NEW_',@id"/>
</xsl:attribute>
<xsl:copy>
<xsl:apply-templates select="node()"/>
</xsl:copy>
</product>
</xsl:template>
但是,使用上面的转换,我得到以下 XML 输出:
<?xml version="1.0" encoding="utf-8"?>
<products author="Jesper">
<product id="p1">
<name>Delta</name>
<price>800</price>
<stock>4</stock>
<country>Denmark</country>
</product>
<product id="NEW_p1"><product>
<name>Delta</name>
<price>800</price>
<stock>4</stock>
<country>Denmark</country>
</product></product>
</products>
如您所见,添加了 product 元素,同时我声明了一个具有新 @id 值的新 product 元素。因为我用来处理子节点,所以我相信这会再次处理 product 元素。
此外,我需要帮助更新子节点的值(在每个值前添加 'NEW_')。在这个网站上搜索大量问题,我相信我需要一个这样的模板:
<xsl:template match="*">
<xsl:element name ="{local-name()}">
<!--for all attributes-->
<xsl:copy-of select ="@*"/>
<xsl:value-of select = "."/>
</xsl:element>
</xsl:template>
提前感谢您 suggestions/ideas 我的问题。
已更新 感谢@Mathias 对我最初问题的回答。提供的答案又带来了一个涉及递归到 XML 结构的更深层次的问题。
输入XML文件:
<products author="Jesper">
<product id="p1">
<name>Delta
<innerName>MiddleDelta
<baseName>FinalDelta</baseName>
</innerName>
</name>
<price>800</price>
<stock>4</stock>
<country>Denmark
<city>Copenhagen</city>
</country>
</product>
</products>
更新后的期望输出文件如下:
<?xml version="1.0" encoding="utf-8"?>
<products author="Jesper">
<product id="p1">
<name>Delta
<innerName>MiddleDelta
<baseName>FinalDelta</baseName>
</innerName>
</name>
<price>800</price>
<stock>4</stock>
<country>Denmark
<city>Copenhagen</city>
</country>
</product>
<product id="NEW_p1">
<name>NEW_Delta
<innerName>NEW_MiddleDelta
<baseName>NEW_FinalDelta</baseName>
</innerName>
</name>
<price>NEW_800</price>
<stock>NEW_4</stock>
<country>NEW_Denmark
<city>NEW_Copenhagen</city>
</country>
</product>
</products>
我只能猜测使用模板会起作用,因为看到每个节点都有不同级别的子节点。在此先感谢您ideas/suggestions。
你已经不远了,但主要有两个问题:
- 您在第二个文字
product
元素中使用xsl:copy
,这会在输出 中产生一个额外的 - 您需要一种方法来找到
product
元素的所有子元素,再次输出它们并将 "NEW_" 添加到它们的文本内容中。
product
元素
您确定 version
属性应设置为“2.0”吗?另外,我不确定这个练习的要点是什么...
XSLT 样式表
<?xml version="1.0" encoding="utf-8"?>
<xsl:stylesheet version="2.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="xml" indent="yes"/>
<xsl:template match="@*|node()">
<xsl:copy>
<xsl:apply-templates select="@*|node()"/>
</xsl:copy>
</xsl:template>
<xsl:template match ="product">
<xsl:copy>
<xsl:apply-templates select="@*|node()"/>
</xsl:copy>
<product id="{concat('NEW_',@id)}">
<xsl:for-each select="*">
<xsl:copy>
<xsl:value-of select="concat('NEW_',.)"/>
</xsl:copy>
</xsl:for-each>
</product>
</xsl:template>
</xsl:stylesheet>
XML输出
<?xml version="1.0" encoding="UTF-8"?>
<products author="Jesper">
<product id="p1">
<name>Delta</name>
<price>800</price>
<stock>4</stock>
<country>Denmark</country>
</product>
<product id="NEW_p1">
<name>NEW_Delta</name>
<price>NEW_800</price>
<stock>NEW_4</stock>
<country>NEW_Denmark</country>
</product>
</products>
这是对您更新后的问题的回应(恕我直言,应该将其作为一个新问题提出):
XSLT 1.0
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>
<xsl:strip-space elements="*"/>
<xsl:template match="/products">
<xsl:copy>
<xsl:copy-of select="product"/>
<xsl:apply-templates select="product"/>
</xsl:copy>
</xsl:template>
<!-- modified identity transform -->
<xsl:template match="*">
<xsl:copy>
<xsl:apply-templates select="@*|node()"/>
</xsl:copy>
</xsl:template>
<xsl:template match="@*">
<xsl:attribute name="{name()}">
<xsl:value-of select="concat('NEW_', .)"/>
</xsl:attribute>
</xsl:template>
<xsl:template match="text()">
<xsl:value-of select="concat('NEW_', .)"/>
</xsl:template>
</xsl:stylesheet>