c# - XSLT 转换 : Updating nodes of output xml

标签 c# xml xslt xpath

我有以下场景,我正在努力解决。我不完全确定我的解决方案首先是否正确(但它可以使用 XPath 和 C# 代码)。在 xslt 中也复制了几乎相似的逻辑。

注意事项: XML 是来自第三方的输入。我不能改变它的结构。

所以我输入的xml是这样的

<?xml version="1.0" encoding="utf-8"?>
<Root>
  <Houses>
    <House id="0" address="House1" area="XX"/>
    <House id="1" address="House2" area="XX"/>
    <House id="0" address="House1" area="YY"/>
    <House id="1" address="House2" area="YY"/>
  </Houses>
  <VisitModule>
    <VisitedBy personID="ABC">
      <VisitedArea id="XX">
        <VisitedHouse houseID="0" isVisited="false" />
        <VisitedHouse houseID="1" isVisited="false" />
      </VisitedArea>
    </VisitedBy>
    <VisitedBy personID="XYZ">
      <VisitedArea id="XX">
        <VisitedHouse houseID="0" isVisited="true" />
        <VisitedHouse houseID="1" isVisited="false" />
      </VisitedArea>
      <VisitedArea id="YY">
        <VisitedHouse houseID="0" isVisited="false" />
        <VisitedHouse houseID="1" isVisited="false" />
      </VisitedArea>
    </VisitedBy>
  </VisitModule>
</Root>

我希望实现的是,如果任何人访问过这所房子,那么该房子将被标记为已访问过,否则不会。

我需要的输出 xml 有点像下面,

<?xml version="1.0" encoding="utf-8"?>
<Root>
  <VisitedArea id="XX">
    <!--If covered by any person, its covered=true.-->
    <House id ="0" covered="true" />
    <House id ="1" covered="false" />
  </VisitedArea>
  <VisitedArea id="YY">
    <House id ="0" covered="false" />
    <House id ="1" covered="false" />
  </VisitedArea>
</Root>

我已经将其作为 XML 遍历的一部分。但它的速度慢得令人难以置信(因为当前输入的 XML 非常庞大)。因此希望通过 XSLT 来完成,这应该会更快。

我的思路是,遍历每个house节点,根据area和id找到匹配的houseID,然后进行计算。

我已经在 XSLT 中完成了大部分工作,除了我需要更新现有节点数据的地方,比如 XX 区域的房子 1,以前没有访问过但现在我找到了一个访问过这个节点的人房子,所以我现在需要将该节点设置为

covered=true

我找不到任何指向编辑当前正在转换的文档的内容。我并不是说我的方法 100% 正确,所以我也对其他想法持开放态度。但我认为使用 XSLT 将使我的工作在维护方面比实际代码更容易,因此非常希望在 XSLT 中完成它。

提前致谢。 :)

最佳答案

这是一种使用 XSLT 的可能解决方案。它的工作原理是获取所有区域的不同列表,然后使用 Houses 子元素下的房屋列表,检查每个房屋以查看在该房屋所在的访问区域下是否有匹配的节点covered 以获得 covered 属性的正确值。

<?xml version="1.0" encoding="utf-8"?>
<xsl:stylesheet version="1.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"/>
    <!-- get distinct list of areas -->
    <xsl:template match="/">
      <Root>
        <xsl:for-each select="//House/@area[not(.=following::House/@area)]">
          <xsl:call-template name="areaTemplate">
            <xsl:with-param name="areaCode" select="." />
          </xsl:call-template>
        </xsl:for-each>
      </Root>
    </xsl:template>

    <!-- for each area check each house to see if covered -->
    <xsl:template name="areaTemplate">
      <xsl:param name="areaCode" />
      <VisitedArea id="{$areaCode}">
      <xsl:for-each select="//House[@area=$areaCode]">
        <xsl:variable name ="houseId" select="@id" />
        <xsl:variable name ="covered" select="boolean(//VisitedArea[@id=$areaCode]/VisitedHouse[@isVisited='true' and @houseID=$houseId])" />
        <House id="{$houseId}" covered="{$covered}" />
      </xsl:for-each>
      </VisitedArea>
  </xsl:template>

</xsl:stylesheet>

给定示例输入 XML,输出应该是这样的:

<?xml version="1.0" encoding="utf-8"?>
<Root>
  <VisitedArea id="XX">
    <House id="0" covered="true" />
    <House id="1" covered="false" />
  </VisitedArea>
  <VisitedArea id="YY">
    <House id="0" covered="false" />
    <House id="1" covered="false" />
  </VisitedArea>
</Root>

我还没有测试它的性能;另一种方法可能是将您的输入 xml 反序列化为类,以将它们操作到必要的组中,然后序列化回您的目标 xml。

关于c# - XSLT 转换 : Updating nodes of output xml,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/30303396/

相关文章:

c# MySql.Data 未加载

XML 和 XSLT 保留 CDATA

xml - XSL 根据子项向父项添加属性

xslt - 不基于其父节点选择的节点的索引

c# - 如何在 c# winforms 应用程序中获取列表框项目的 "key"?

c# - 如何将 C# 代码转换为 C++

c# - 这是我的 child 吗?

java - 将 XMl 文件转换为字符串

xml - 使用 Ruby 和 Hpricot 将 xml 转换为 yaml - 这里出了什么问题?

java - 使用已编译的 XSL 转换