xml - 在 XSLT 1.0 中使用计时逻辑的 JSON 输出
问题描述
我正在尝试读取 XML 中的消息并以 JSON 格式输出。每条消息还有一个定时逻辑来显示和隐藏消息。
下面是我的 XML,每条消息都有开始和结束日期/时间
<activeMessage>
<message>
<messageText>test message 1</messageText>
<displayScheduleContainer>
<startDate>17/05/2019</startDate>
<startTimeHrs>12</startTimeHrs>
<startTimeMins>00</startTimeMins>
<noEndDate/>
<endDate>17/05/2019</endDate>
<endTimeHrs>23</endTimeHrs>
<endTimeMins>59</endTimeMins>
</displayScheduleContainer>
</message>
下面是我的 XSL
<xsl:for-each select="xalan:nodeset($messageData)/activeMessage/message">
<xsl:variable name="messageInDateTime">
<xsl:call-template name="dateLessThanTemplate">
<xsl:with-param name="startDateTime" select="concat(displayScheduleContainer/startDate, ' ', displayScheduleContainer/startTimeHrs, ':', displayScheduleContainer/startTimeMins)" />
<xsl:with-param name="endDateTime" select="concat(displayScheduleContainer/endDate, ' ', displayScheduleContainer/endTimeHrs, ':', displayScheduleContainer/endTimeMins)" />
</xsl:call-template>
</xsl:variable>
<xsl:if test="$messageInDateTime = 'true'">
<xsl:choose>
<xsl:when test="position()=1">
<xsl:call-template name="singleMessageJSON" />
</xsl:when>
<xsl:otherwise>
<xsl:call-template name="multiMessageJSON" />
</xsl:otherwise>
</xsl:choose>
</xsl:if>
</xsl:for-each>
解决方案
当您为 XSLT 3 添加标签时,我认为您可以摆脱任何 Java 调用,只需使用例如
<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:xs="http://www.w3.org/2001/XMLSchema"
xmlns:mf="http://example.com/mf"
exclude-result-prefixes="#all"
version="3.0">
<xsl:function name="mf:get-dateTime" as="xs:dateTime?">
<xsl:param name="date" as="xs:string"/>
<xsl:param name="hours" as="xs:string"/>
<xsl:param name="minutes" as="xs:string"/>
<xsl:try
select="xs:dateTime(replace($date, '([0-9]{2})/([0-9]{2})/([0-9]{4})', '$3-$2-$1') || 'T' || $hours || ':' || $minutes || ':00')">
<xsl:catch select="()"/>
</xsl:try>
</xsl:function>
<xsl:param name="now" select="current-dateTime()"/>
<xsl:output method="json" indent="yes"/>
<xsl:template match="activeMessage">
<xsl:sequence
select="map {
'message' : array {
message[mf:get-dateTime(displayScheduleContainer/startDate, displayScheduleContainer/startTimeHrs, displayScheduleContainer/startTimeMins) lt $now
and
mf:get-dateTime(displayScheduleContainer/endDate, displayScheduleContainer/endTimeHrs, displayScheduleContainer/endTimeMins) gt $now]/messageText[normalize-space()]/string()
}
}"/>
</xsl:template>
</xsl:stylesheet>
在 XSLT 3 和 Saxon 9.8 或更高版本中。我不确定您想要的确切 JSON 输出格式,但 XSLT 3 的主要优点是您可以在 XSLT/XPath 中构造映射和数组,如果需要,它们会被序列化为 JSON,而不必担心在正确的位置输出逗号.
https://xsltfiddle.liberty-development.net/bnnZWx/1
将日期的两个比较移到另一个函数中可能会更好,因为该谓词变得更具可读性:
<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:xs="http://www.w3.org/2001/XMLSchema"
xmlns:mf="http://example.com/mf"
exclude-result-prefixes="#all"
version="3.0">
<xsl:function name="mf:get-dateTime" as="xs:dateTime?">
<xsl:param name="date" as="xs:string"/>
<xsl:param name="hours" as="xs:string"/>
<xsl:param name="minutes" as="xs:string"/>
<xsl:try
select="xs:dateTime(replace($date, '([0-9]{2})/([0-9]{2})/([0-9]{4})', '$3-$2-$1') || 'T' || $hours || ':' || $minutes || ':00')">
<xsl:catch select="()"/>
</xsl:try>
</xsl:function>
<xsl:function name="mf:compare-dates" as="xs:boolean">
<xsl:param name="schedule" as="element(displayScheduleContainer)"/>
<xsl:sequence
select="mf:get-dateTime($schedule/startDate, $schedule/startTimeHrs, $schedule/startTimeMins) lt $now
and
mf:get-dateTime($schedule/endDate, $schedule/endTimeHrs, $schedule/endTimeMins) gt $now"/>
</xsl:function>
<xsl:param name="now" select="current-dateTime()"/>
<xsl:output method="json" indent="yes"/>
<xsl:template match="activeMessage">
<xsl:sequence
select="map {
'message' : array {
message[mf:compare-dates(displayScheduleContainer)]/messageText[normalize-space()]/string()
}
}"/>
</xsl:template>
</xsl:stylesheet>
推荐阅读
- python - Python 2.7 版没有模块 http.client
- python - 如何将值插入 SQLAlchemy TIMESTAMP 列
- websphere-liberty - 我缺少 wlp-kernel 2019.3.0.0
- python-3.x - 编号的工作日:找到第 k 个整数
- c# - 在解析具有 anyOf 属性下的多个引用的 json 模式时,输出中仅返回一个引用(最后一个)
- javascript - 使用 Jasmine 对过滤器组件进行单元测试
- jquery - 在href jquery中使用不同域的目标空白
- c# - 如何从类库项目和预编译视图创建 NuGet 包?
- excel - 如何从逗号分隔的单元格中查找多个值并在 excel 中平均结果?
- android - 不使用 volley 无法触发 mysql 查询