Xml 根据无周末的订单日期计算交货日期

Xml 根据无周末的订单日期计算交货日期,xml,xslt,xslt-1.0,Xml,Xslt,Xslt 1.0,我正试图找到一个基于订单日期(deliverydate等于订单日期后3天)计算交货日期的解决方案。但是,需要考虑周末 <order> <orderdate>30/11/2016</orderdate> </order> 30/11/2016 因此,交货日期必须为2016年12月5日 有人在这件事上提供帮助吗 任何帮助都将受到感激 在纯XSLT 1.0中向日期添加N个工作日/工作日 我一直在犹豫是计算目标日期,还是使用递归。最后,我决定使

我正试图找到一个基于订单日期(deliverydate等于订单日期后3天)计算交货日期的解决方案。但是,需要考虑周末

<order>
   <orderdate>30/11/2016</orderdate>
</order>

30/11/2016
因此,交货日期必须为2016年12月5日

有人在这件事上提供帮助吗

任何帮助都将受到感激

在纯XSLT 1.0中向日期添加N个工作日/工作日 我一直在犹豫是计算目标日期,还是使用递归。最后,我决定使用递归,因为这也是唯一的考虑假期的方式。

输入XML

<orders>
    <order>
        <order-date>2016-11-21</order-date>
    </order>
    <order>
        <order-date>2016-11-22</order-date>
    </order>
    <order>
        <order-date>2016-11-23</order-date>
    </order>
    <order>
        <order-date>2016-11-24</order-date>
    </order>
    <order>
        <order-date>2016-11-25</order-date>
    </order>
    <order>
        <order-date>2016-11-26</order-date>
    </order>
    <order>
        <order-date>2016-11-27</order-date>
    </order>
</orders>

2016-11-21
2016-11-22
2016-11-23
2016-11-24
2016-11-25
2016-11-26
2016-11-27
XSLT1.0

<xsl:stylesheet version="1.0" 
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>
<xsl:strip-space elements="*"/>

<!-- identity transform -->
<xsl:template match="@*|node()">
    <xsl:copy>
        <xsl:apply-templates select="@*|node()"/>
    </xsl:copy>
</xsl:template>

<xsl:template match="order">
    <xsl:copy>
        <xsl:copy-of select="order-date"/>
        <delivery-date>
            <xsl:call-template name="add-work-days">
                <xsl:with-param name="start-JDN">
                    <xsl:call-template name="JDN">
                        <xsl:with-param name="date" select="order-date" />
                    </xsl:call-template>
                </xsl:with-param>
                <xsl:with-param name="add-days" select="3" />
            </xsl:call-template>
        </delivery-date>
    </xsl:copy>
</xsl:template> 

<xsl:template name="add-work-days">
    <xsl:param name="start-JDN"/>
    <xsl:param name="add-days"/>

    <xsl:variable name="day-of-week" select="$start-JDN mod 7"/>
    <!--  Monday = 0 ... Saturday = 5, Sunday = 6 -->
    <xsl:choose>
        <xsl:when test="$day-of-week >= 5">
            <xsl:call-template name="add-work-days">
                <xsl:with-param name="start-JDN" select="$start-JDN + 1" />
                <xsl:with-param name="add-days" select="$add-days" />
            </xsl:call-template>
        </xsl:when>
        <xsl:when test="$add-days > 0">
            <xsl:call-template name="add-work-days">
                <xsl:with-param name="start-JDN" select="$start-JDN + 1" />
                <xsl:with-param name="add-days" select="$add-days - 1" />
            </xsl:call-template>
        </xsl:when>
        <xsl:otherwise>
            <xsl:call-template name="GD">
                <xsl:with-param name="JDN" select="$start-JDN" />
            </xsl:call-template>
        </xsl:otherwise>
    </xsl:choose>
</xsl:template> 

<xsl:template name="JDN">
    <xsl:param name="date"/>
    <xsl:variable name="year" select="substring($date, 1, 4)"/>
    <xsl:variable name="month" select="substring($date, 6, 2)"/>
    <xsl:variable name="day" select="substring($date, 9, 2)"/>
    <xsl:variable name="a" select="floor((14 - $month) div 12)"/>
    <xsl:variable name="y" select="$year + 4800 - $a"/>
    <xsl:variable name="m" select="$month + 12*$a - 3"/>
    <xsl:value-of select="$day + floor((153*$m + 2) div 5) + 365*$y + floor($y div 4) - floor($y div 100) + floor($y div 400) - 32045" />
</xsl:template> 

<xsl:template name="GD">
    <xsl:param name="JDN"/>
    <xsl:variable name="f" select="$JDN + 1401 + floor((floor((4 * $JDN + 274277) div 146097) * 3) div 4) - 38"/>
    <xsl:variable name="e" select="4*$f + 3"/>
    <xsl:variable name="g" select="floor(($e mod 1461) div 4)"/>
    <xsl:variable name="h" select="5*$g + 2"/>
    <xsl:variable name="D" select="floor(($h mod 153) div 5 ) + 1"/>
    <xsl:variable name="M" select="(floor($h div 153) + 2) mod 12 + 1"/>
    <xsl:variable name="Y" select="floor($e div 1461) - 4716 + floor((14 - $M) div 12)"/>

    <xsl:value-of select="$Y" />    
    <xsl:value-of select="format-number($M, '-00')"/>
    <xsl:value-of select="format-number($D, '-00')"/>
</xsl:template>     

</xsl:stylesheet>

结果

<?xml version="1.0" encoding="UTF-8"?>
<orders>
   <order>
      <order-date>2016-11-21</order-date>
      <delivery-date>2016-11-24</delivery-date>
   </order>
   <order>
      <order-date>2016-11-22</order-date>
      <delivery-date>2016-11-25</delivery-date>
   </order>
   <order>
      <order-date>2016-11-23</order-date>
      <delivery-date>2016-11-28</delivery-date>
   </order>
   <order>
      <order-date>2016-11-24</order-date>
      <delivery-date>2016-11-29</delivery-date>
   </order>
   <order>
      <order-date>2016-11-25</order-date>
      <delivery-date>2016-11-30</delivery-date>
   </order>
   <order>
      <order-date>2016-11-26</order-date>
      <delivery-date>2016-12-01</delivery-date>
   </order>
   <order>
      <order-date>2016-11-27</order-date>
      <delivery-date>2016-12-01</delivery-date>
   </order>
</orders>

2016-11-21
2016-11-24
2016-11-22
2016-11-25
2016-11-23
2016-11-28
2016-11-24
2016-11-29
2016-11-25
2016-11-30
2016-11-26
2016-12-01
2016-11-27
2016-12-01

注意

为了使其在一般情况下有用,我对输入和输出都使用了标准的YYYY-MM-DD日期格式。如果输入使用另一种格式,则必须调整
JDN
模板的开头,以便它正确提取日期组件。如果希望输出使用另一种格式,则需要修改
GD
模板的结尾,以便它按照要求的顺序组合最终日期。

在纯XSLT 1.0中为日期添加N个工作日/营业日 我一直在犹豫是计算目标日期,还是使用递归。最后,我决定使用递归,因为这也是唯一的考虑假期的方式。

输入XML

<orders>
    <order>
        <order-date>2016-11-21</order-date>
    </order>
    <order>
        <order-date>2016-11-22</order-date>
    </order>
    <order>
        <order-date>2016-11-23</order-date>
    </order>
    <order>
        <order-date>2016-11-24</order-date>
    </order>
    <order>
        <order-date>2016-11-25</order-date>
    </order>
    <order>
        <order-date>2016-11-26</order-date>
    </order>
    <order>
        <order-date>2016-11-27</order-date>
    </order>
</orders>

2016-11-21
2016-11-22
2016-11-23
2016-11-24
2016-11-25
2016-11-26
2016-11-27
XSLT1.0

<xsl:stylesheet version="1.0" 
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>
<xsl:strip-space elements="*"/>

<!-- identity transform -->
<xsl:template match="@*|node()">
    <xsl:copy>
        <xsl:apply-templates select="@*|node()"/>
    </xsl:copy>
</xsl:template>

<xsl:template match="order">
    <xsl:copy>
        <xsl:copy-of select="order-date"/>
        <delivery-date>
            <xsl:call-template name="add-work-days">
                <xsl:with-param name="start-JDN">
                    <xsl:call-template name="JDN">
                        <xsl:with-param name="date" select="order-date" />
                    </xsl:call-template>
                </xsl:with-param>
                <xsl:with-param name="add-days" select="3" />
            </xsl:call-template>
        </delivery-date>
    </xsl:copy>
</xsl:template> 

<xsl:template name="add-work-days">
    <xsl:param name="start-JDN"/>
    <xsl:param name="add-days"/>

    <xsl:variable name="day-of-week" select="$start-JDN mod 7"/>
    <!--  Monday = 0 ... Saturday = 5, Sunday = 6 -->
    <xsl:choose>
        <xsl:when test="$day-of-week >= 5">
            <xsl:call-template name="add-work-days">
                <xsl:with-param name="start-JDN" select="$start-JDN + 1" />
                <xsl:with-param name="add-days" select="$add-days" />
            </xsl:call-template>
        </xsl:when>
        <xsl:when test="$add-days > 0">
            <xsl:call-template name="add-work-days">
                <xsl:with-param name="start-JDN" select="$start-JDN + 1" />
                <xsl:with-param name="add-days" select="$add-days - 1" />
            </xsl:call-template>
        </xsl:when>
        <xsl:otherwise>
            <xsl:call-template name="GD">
                <xsl:with-param name="JDN" select="$start-JDN" />
            </xsl:call-template>
        </xsl:otherwise>
    </xsl:choose>
</xsl:template> 

<xsl:template name="JDN">
    <xsl:param name="date"/>
    <xsl:variable name="year" select="substring($date, 1, 4)"/>
    <xsl:variable name="month" select="substring($date, 6, 2)"/>
    <xsl:variable name="day" select="substring($date, 9, 2)"/>
    <xsl:variable name="a" select="floor((14 - $month) div 12)"/>
    <xsl:variable name="y" select="$year + 4800 - $a"/>
    <xsl:variable name="m" select="$month + 12*$a - 3"/>
    <xsl:value-of select="$day + floor((153*$m + 2) div 5) + 365*$y + floor($y div 4) - floor($y div 100) + floor($y div 400) - 32045" />
</xsl:template> 

<xsl:template name="GD">
    <xsl:param name="JDN"/>
    <xsl:variable name="f" select="$JDN + 1401 + floor((floor((4 * $JDN + 274277) div 146097) * 3) div 4) - 38"/>
    <xsl:variable name="e" select="4*$f + 3"/>
    <xsl:variable name="g" select="floor(($e mod 1461) div 4)"/>
    <xsl:variable name="h" select="5*$g + 2"/>
    <xsl:variable name="D" select="floor(($h mod 153) div 5 ) + 1"/>
    <xsl:variable name="M" select="(floor($h div 153) + 2) mod 12 + 1"/>
    <xsl:variable name="Y" select="floor($e div 1461) - 4716 + floor((14 - $M) div 12)"/>

    <xsl:value-of select="$Y" />    
    <xsl:value-of select="format-number($M, '-00')"/>
    <xsl:value-of select="format-number($D, '-00')"/>
</xsl:template>     

</xsl:stylesheet>

结果

<?xml version="1.0" encoding="UTF-8"?>
<orders>
   <order>
      <order-date>2016-11-21</order-date>
      <delivery-date>2016-11-24</delivery-date>
   </order>
   <order>
      <order-date>2016-11-22</order-date>
      <delivery-date>2016-11-25</delivery-date>
   </order>
   <order>
      <order-date>2016-11-23</order-date>
      <delivery-date>2016-11-28</delivery-date>
   </order>
   <order>
      <order-date>2016-11-24</order-date>
      <delivery-date>2016-11-29</delivery-date>
   </order>
   <order>
      <order-date>2016-11-25</order-date>
      <delivery-date>2016-11-30</delivery-date>
   </order>
   <order>
      <order-date>2016-11-26</order-date>
      <delivery-date>2016-12-01</delivery-date>
   </order>
   <order>
      <order-date>2016-11-27</order-date>
      <delivery-date>2016-12-01</delivery-date>
   </order>
</orders>

2016-11-21
2016-11-24
2016-11-22
2016-11-25
2016-11-23
2016-11-28
2016-11-24
2016-11-29
2016-11-25
2016-11-30
2016-11-26
2016-12-01
2016-11-27
2016-12-01

注意


为了使其在一般情况下有用,我对输入和输出都使用了标准的YYYY-MM-DD日期格式。如果输入使用另一种格式,则必须调整
JDN
模板的开头,以便它正确提取日期组件。如果您希望输出使用另一种格式,则需要修改
GD
模板的结尾,以便它按照要求的顺序组合最终日期。

请详细解释“需要考虑周末”的含义。您需要计算交付日期,不包括周末。例如,如果订单日期为2016年12月2日(星期五),则交货日期必须在3个工作日之后,即2016年12月7日(星期三)。您不断给我举例,而不是解释规则。我甚至不知道你们公司的哪一天是周末。周六和周日是周末。交付日期2的计算不包括周末。必须始终在订单日期后3个工作日内。请详细说明“需要考虑周末”的含义。您需要计算不包括周末在内的交货日期。例如,如果订单日期为2016年12月2日(星期五),则交货日期必须在3个工作日之后,即2016年12月7日(星期三)。您不断给我举例,而不是解释规则。我甚至不知道你们公司的哪一天是周末。周六和周日是周末。交付日期2的计算不包括周末。必须始终在订单日期后3个工作日内完成,例如
2016-02-282016-03-02
,因为2016年是闰年,有29个days@kokos你为什么不试试看呢。我试过了,它的计算2016-03-032016-02-28是星期天,所以计数只在星期一开始,结果是下一个星期四。如果你从2016年11月27日星期日开始,你会看到同样的事情:结果将是2016年12月1日星期四。这与闰年无关。例如
2016-02-282016-03-02
,因为2016年是闰年,有29个闰年days@kokos你为什么不自己试试看呢。我已经试过了,它的名字是2016-03-032