xml - 累加器不适用于 Saxon PE 9-9-1-5 Java 的当前文档
问题描述
我有一个输入 XML:
<?xml version='1.0' encoding='UTF-8'?>
<MessageFormat name='TransportEvent' version='2.02'>
<StructFormat name='TransportEventHeader' >
<TagField type='String' value='H0 '/>
<FieldFormat name='NotificationTriggeringOU' length='10'/>
<FieldFormat name='NotificationTriggeringIT' length='8'/>
<FieldFormat name='NotificationReference' length='1'/>
<FieldFormat name='NotificationCode' length='3'/>
<FieldFormat name='NotificationType' length='8'/>
<FieldFormat name='NotificationStatus' length='1'/>
<FieldFormat name='NotificationTripType' length='1'/>
<FieldFormat name='ProducingRailwayUndertaking' length='4'/>
<FieldFormat name='ExternalPartner' length='35'/>
<FieldFormat name='ActualNumberOfWagons' length='3' />
<FieldFormat name='ProcessingTime' length='26'/>
<StructFormat name='NotificationFunctionalClassification'>
<FieldFormat name='OrderRelevant' length='1'/>
<FieldFormat name='TimetableRelevant' length='1'/>
<FieldFormat name='CapacityRelevant' length='1'/>
<FieldFormat name='IntermodalRelevant' length='1'/>
<FieldFormat name='XrailRelevant' length='1'/>
<FieldFormat name='NotificationLocationRelevant' length='1'/>
</StructFormat>
<FieldFormat name='Reserve' length='1'/>
</StructFormat>
<StructFormat name='NotificationLocation' >
<TagField type='String' value='M1 '/>
<StructFormat name='CurrentLocation'>
<FieldFormat name='CurrentLocationRL100' length='5'/>
<FieldFormat name='CurrentLocationLocationType' length='1'/>
<FieldFormat name='CurrentUICRailAuthorityNumber' length='6'/>
<FieldFormat name='CurrentNetworkLocationNumber' length='6'/>
<FieldFormat name='CurrentLocationSatelliteNumber' length='2'/>
<FieldFormat name='CurrentFreightCarLocationNumber' length='4'/>
</StructFormat>
<StructFormat name='NextLocation'>
<FieldFormat name='NextLocationRL100' length='5'/>
<FieldFormat name='NextLocationLocationType' length='1'/>
<FieldFormat name='NextUICRailAuthorityNumber' length='6'/>
<FieldFormat name='NextNetworkLocationNumber' length='6'/>
<FieldFormat name='NextLocationSatelliteNumber' length='2'/>
<FieldFormat name='NextFreightCarLocationNumber' length='4'/>
</StructFormat>
</StructFormat>
<StructFormat name='NotificationTime' >
<TagField type='String' value='M2 '/>
<FieldFormat name='ActualTime' length='18'/>
</StructFormat>
<StructFormat name='Trip' >
<TagField type='String' value='Z1 '/>
<FieldFormat name='TripNumber' length='6'/>
<FieldFormat name='RegionNetz' length='2'/>
<FieldFormat name='NationalProductionDate' length='10'/>
<FieldFormat name='TrainTypeMainNumber' length='2'/>
<FieldFormat name='TrainTypeSubNumber' length='1'/>
<FieldFormat name='DepartureStationRL100' length='5'/>
<FieldFormat name='DepartureStationUICRailAuthority' length='6'/>
<FieldFormat name='DepartureStationNetworkLocation' length='6'/>
<FieldFormat name='TargetTime' length='18'/>
<FieldFormat name='RelativeTime' length='5' />
</StructFormat>
<StructFormat name='HandoverTakeover' >
<TagField type='String' value='U1 '/>
<FieldFormat name='HandoverTakeoverFlag' length='1'/>
<FieldFormat name='ConsigningRU' length='4'/>
<FieldFormat name='AcceptingRU' length='4'/>
<FieldFormat name='UICBorderCode' length='3'/>
</StructFormat>
</MessageFormat>
在我的 xslt 模板中,我想直接在 MessageFormat 下计算每个 StructFormat 的 (FieldFormat/@length + string-length(TagField/@value):
<?xml version="1.0" encoding="UTF-8"?>
<?altova_samplexml format.xml?>
<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform" xmlns:xs="http://www.w3.org/2001/XMLSchema" xmlns:fn="http://www.w3.org/2005/xpath-functions" xmlns:math="http://www.w3.org/2005/xpath-functions/math" xmlns:array="http://www.w3.org/2005/xpath-functions/array" xmlns:map="http://www.w3.org/2005/xpath-functions/map" xmlns:xhtml="http://www.w3.org/1999/xhtml" xmlns:err="http://www.w3.org/2005/xqt-errors" xmlns:saxon="http://saxon.sf.net/" exclude-result-prefixes="array fn map math xhtml xs err" version="3.0">
<xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>
<xsl:variable name="msg">H0 EVU_DBSRD PVG W14 PVGERL I 0142019-12-08-07.46.52.436704NNJNNJNM1 80270611 04401 M2 08.12.201907:50:00Z1 BAU1 08.12.2019 08.12.201907:50:00 0000R1 00131806940728280201912063946287201912061215058175346965 000NNJNNN 80270611 04401R1 00231806940476880201912063946287201912061215059494346965 000NNJNNN 80270611 04401R1 00331806948174180201912063946287201912061215050425346965 000NNJNNN 80270611 04401R1 00431806948098280201912063946287201912061215051864346965 000NNJNNN 80270611 04401</xsl:variable>
<xsl:accumulator name="position-count" as="xs:double" initial-value="1" streamable="no">
<xsl:accumulator-rule match="/MessageFormat/StructFormat" phase="end" select="$value +sum(.//FieldFormat/@length) + string-length(./TagField/@value)" />
</xsl:accumulator>
<xsl:template match="/">
<output>
<xsl:for-each select="/MessageFormat/StructFormat">
<xsl:element name="{./@name}">
<xsl:attribute name="start" select="fn:accumulator-before('position-count')"/>
<xsl:attribute name="end" select="fn:accumulator-after('position-count')" />
</xsl:element>
</xsl:for-each>
</output>
</xsl:template>
</xsl:stylesheet>
当我使用 XML Spy Professional 2020 SP1 运行它时,我得到了输出:
<?xml version="1.0" encoding="UTF-8"?>
<output xmlns:saxon="http://saxon.sf.net/">
<TransportEventHeader start="1" end="111"/>
<NotificationLocation start="111" end="162"/>
<NotificationTime start="162" end="183"/>
<Trip start="183" end="247"/>
<HandoverTakeover start="247" end="262"/>
</output>
但是当我用 Saxon 9-9-1-5 Java 运行它时,java -cp /d/SaxonPE9-9-1-5J/saxon9pe.jar net.sf.saxon.Transform -s:format.xml -xsl:accumulator.xslt -o:output.xml
我得到了错误:
在 accumulator.xslt 的第 13 行第 83 列的 xsl:attribute/@select 中评估 (fn:accumulator-before(...)) 时出错:XTDE3362:累加器位置计数不适用于未知调用者调用的当前文档(类net.sf.saxon.expr.instruct.ForEach) at file:/D:/xslt/accumulator.xslt#11 在 accumulator.xslt 第 9 行带有 match="/" 的模板规则中,累加器位置计数不适用于当前文件
我的蓄能器出了什么问题?为什么它适用于 XML Spy,而不适用于 Saxon 9.9?
实际上我想在我的累加器中添加一个条件来检查TagField/@value
变量中的值是否出现$msg
在正确的位置。
例如 for StructFormat[@name='TransportEventHeader']
,$msg 中的前 3 个字符是“H0”,所以它匹配StructFormat[@name='TransportEventHeader']/TagField/@value
,在这种情况下应该将长度添加到我的累加器中,如果没有,则不应该添加。我不知道如何在累加器中实现它。
解决方案
规则在这里:https ://www.w3.org/TR/xslt-30/#applicability-of-accumulators
规则 5 说:对于包含在初始匹配选择中提供的节点的文档,适用的累加器是由初始模式的 xsl:mode 声明确定的累加器。这意味着在没有 xsl:mode 声明的情况下,没有累加器适用。
所以你需要添加
<xsl:mode use-accumulators="position-count"/>
规范中这些规则的原因是(a)对于流式传输,如果不打算在该文档上使用流式文档,那么维护一个累加器的成本很高,并且(b)尽管不存在相同的开销对于非流式文档,因为可以在第一次使用时评估累加器,所以认为无论文档是流式还是非流式,都需要具有相同的规则。
Altova 没有实现流式传输,因此他们可能认为(并且有一些理由,我同意)实施规则需要付出很多努力,但对用户几乎没有价值。
推荐阅读
- list - 如何在 Lua 中有嵌套列表
- php - 如何使用锚标记将两个相关表中的两个值传递到另一个页面
- java - 在java中使用HSB创建平滑渐变
- ajax - 如何使每个 ID 或编号的选择字段(年份列表)唯一并从下拉列表中删除选定的年份?
- python - 如何将数据框中特定行的“零”替换为“一”
- c - 如何将(枚举类型类型,...)传递给另一个需要(枚举类型类型,...)的函数?
- css - MUI - 单击文本字段中的任意位置时打开日期选择器
- python - 为什么 arr.flat.base 与 a.ravel().base 不同?
- modbus - Minimalmodbus:在同一个串口上读取多个从机
- excel - 为什么需要激活特定的工作表才能正确运行宏?