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