XSLT转换:更新输出xml的节点



我有以下场景,我正在努力解决。我不完全确定我的解决方案一开始是否正确(但它使用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来完成它,它应该更快。

我使用的想法是,遍历每个房屋节点,根据面积和id找到与其匹配的houseID,然后进行计算。

除了需要更新现有节点的数据,比如XX区域的1号房子,之前没有访问过,但现在我找到了一个访问过这栋房子的person节点,所以我现在需要将该节点设置为

covered=true

我找不到任何指向编辑当前正在转换的文档的内容。我并不是说我的方法100%正确,所以我也愿意接受其他的想法。但是我认为使用XSLT会使我的生活在维护方面比实际的代码容易得多,所以非常希望用XSLT来完成它。

提前感谢。:)

下面是使用XSLT的一种可能的解决方案。它的工作原理是获取所有区域的不同列表,然后使用Houses子元素下的房屋列表,检查每个区域,看看是否有任何节点匹配该房屋覆盖的访问区域,以获得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。

最新更新