通过XSLT格式化XML中的date

当我使用XML序列化器来序列化一个DateTime ,它被写成以下格式:

 <Date>2007-11-14T12:01:00</Date> 

当通过XSLT样式表传递这个输出HTML时,我该如何格式化呢? 在大多数情况下,我只需要date,当我需要的时候,我当然不希望在那里的“有趣的T”。

这里有几个你可以使用的1.0模板:

 <xsl:template name="formatDate"> <xsl:param name="dateTime" /> <xsl:variable name="date" select="substring-before($dateTime, 'T')" /> <xsl:variable name="year" select="substring-before($date, '-')" /> <xsl:variable name="month" select="substring-before(substring-after($date, '-'), '-')" /> <xsl:variable name="day" select="substring-after(substring-after($date, '-'), '-')" /> <xsl:value-of select="concat($day, ' ', $month, ' ', $year)" /> </xsl:template> <xsl:template name="formatTime"> <xsl:param name="dateTime" /> <xsl:value-of select="substring-after($dateTime, 'T')" /> </xsl:template> 

给他们打电话: –

  <xsl:call-template name="formatDate"> <xsl:with-param name="dateTime" select="xpath" /> </xsl:call-template> 

  <xsl:call-template name="formatTime"> <xsl:with-param name="dateTime" select="xpath" /> </xsl:call-template> 

其中xpath是具有标准date时间格式的元素或属性的path。

在XSLT 1.0中date格式化并不容易。 可能最优雅的方法是在C#中为date格式编写一个简短的XSLT扩展函数。 这是一个例子:

 <?xml version="1.0" encoding="utf-8"?> <xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="1.0" xmlns:msxsl="urn:schemas-microsoft-com:xslt" xmlns:myExtension="urn:myExtension" exclude-result-prefixes="msxsl myExtension"> <xsl:output method="xml" indent="yes"/> <msxsl:script implements-prefix="myExtension" language="C#"> <![CDATA[ public string FormatDateTime(string xsdDateTime, string format) { DateTime date = DateTime.Parse(xsdDateTime); return date.ToString(format); } ]]> </msxsl:script> <xsl:template match="date"> <formattedDate> <xsl:value-of select="myExtension:FormatDateTime(self::node(), 'd')"/> </formattedDate> </xsl:template> </xsl:stylesheet> 

有了这个input文件

 <?xml version="1.0" encoding="utf-8"?> <date>2007-11-14T12:01:00</date> 

你会得到

 <?xml version="1.0" encoding="utf-8"?> <formattedDate>14.11.2007</formattedDate> 

格式化date的函数将date值作为string和格式,如DateTime.ToString方法中所述 。 使用.NET的DateTime结构,您可以免费parsing任意的XSDdate时间值(包括时区说明符),时区计算和本地化输出。

但是,请注意有一个警告(http://support.microsoft.com/kb/316775)与msxml脚本扩展名:每次加载XSLT包含脚本代码的程序集dynamic生成并加载到内存中。; 由于.NET运行时的devise,此程序集不能被卸载。 这就是为什么您必须确保您的XSLT只加载一次(然后caching以供进一步重用)。 在IIS内运行时,这一点尤为重要。

John Workman详细地讨论了这个问题,并在他的博客上给出了这个讨论中的几个解决scheme[1] 。 基本上,parsing每个date组件,并按照您希望的顺序重新组合。 对于你的情况,一个纯粹的XSLT 1.0+版本将是:

 <?xml version="1.0" encoding="utf-8"?> <xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform"> <xsl:template match="date"> <!-- converts FROM <date>2001-12-31T12:00:00</date> TO some new format (DEFINED below) --> <xsl:template name="FormatDate"> <xsl:param name="DateTime" /> <xsl:variable name="year" select="substring($DateTime,1,4)" /> <xsl:variable name="month-temp" select="substring-after($DateTime,'-')" /> <xsl:variable name="month" select="substring-before($month-temp,'-')" /> <xsl:variable name="day-temp" select="substring-after($month-temp,'-')" /> <xsl:variable name="day" select="substring($day-temp,1,2)" /> <xsl:variable name="time" select="substring-after($DateTime,'T')" /> <xsl:variable name="hh" select="substring($time,1,2)" /> <xsl:variable name="mm" select="substring($time,4,2)" /> <xsl:variable name="ss" select="substring($time,7,2)" /> <!-- EUROPEAN FORMAT --> <xsl:value-of select="$day"/> <xsl:value-of select="'.'"/> <!--18.--> <xsl:value-of select="$month"/> <xsl:value-of select="'.'"/> <!--18.03.--> <xsl:value-of select="$year"/> <xsl:value-of select="' '"/> <!--18.03.1976 --> <xsl:value-of select="$hh"/> <xsl:value-of select="':'"/> <!--18.03.1976 13: --> <xsl:value-of select="$mm"/> <xsl:value-of select="':'"/> <!--18.03.1976 13:24 --> <xsl:value-of select="$ss"/> <!--18.03.1976 13:24:55 --> <!-- END: EUROPEAN FORMAT --> </xsl:template> 

另一种格式(replace欧洲格式部分):

 <!-- Long DATE FORMAT --> <xsl:choose> <xsl:when test="$month = '1' or $month= '01'">January</xsl:when> <xsl:when test="$month = '2' or $month= '02'">February</xsl:when> <xsl:when test="$month= '3' or $month= '03'">March</xsl:when> <xsl:when test="$month= '4' or $month= '04'">April</xsl:when> <xsl:when test="$month= '5' or $month= '05'">May</xsl:when> <xsl:when test="$month= '6' or $month= '06'">June</xsl:when> <xsl:when test="$month= '7' or $month= '07'">July</xsl:when> <xsl:when test="$month= '8' or $month= '08'">August</xsl:when> <xsl:when test="$month= '9' or $month= '09'">September</xsl:when> <xsl:when test="$month= '10'">October</xsl:when> <xsl:when test="$month= '11'">November</xsl:when> <xsl:when test="$month= '12'">December</xsl:when> </xsl:choose> <xsl:value-of select="' '"/> <!--January --> <xsl:value-of select="$day"/> <!--January 12 --> <xsl:value-of select="','"/> <!--January 12,--> <xsl:value-of select="' '"/> <!--January 12, --> <xsl:value-of select="$year"/> <!--January 12, 2001--> <!-- END: Long DATE FORMAT --> 

您可以按照您select的任何方式重新组合元素。

[1] http://geekswithblogs.net/workdog/archive/2007/02/08/105858.aspx @@ http://archive.is/4Hjep

道歉评论这个老线程,但对于其他人发现像我一样,如果你使用MS变压器,你也可以使用JavaScript:

声明“msxsl”命名空间:

 xmlns:msxsl="urn:schemas-microsoft-com:xslt" 

为你的脚本声明一个名字空间:

 xmlns:js="urn:custom-javascript" 

(可选)省略输出中的前缀:

 exclude-result-prefixes="msxsl js" 

所以你最终得到一个像这样的xsl声明:

 <xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="1.0" xmlns:msxsl="urn:schemas-microsoft-com:xslt" xmlns:js="urn:custom-javascript" exclude-result-prefixes="msxsl js"> 

将JavaScript写入msxsl:script元素中:

 <msxsl:script language="JavaScript" implements-prefix="js"> <![CDATA[ function javascriptFunction(dateValue){ var date = new Date(dateValue); if(!isNaN(date)) return date.toLocaleString(); return dateValue; } ]]> </msxsl:script> 

调用JavaScript函数(使用XPath语法“。”表示“此节点”):

 <xsl:value-of select="js:javascriptFunction(string(.))"/> 

注意:在编写的时候,似乎没有包含外部js文件(例如jquery库)的(xsl)方法。 这可以通过在转换之前parsingxsl文件服务器端并将js文件内容作为string添加到CDATA部分来完成。 我开始自己走这条路,但得出结论说,如果你需要这个function级别,它可能会更好地放在pipe道的不同部分。

来源: http : //dev.ektron.com/kb_article.aspx?id = 482
ref: http : //www.ibm.com/developerworks/xml/library/x-tipxsltjs/index.html

更正罗伊的职位:function的一天将始终获得月份的价值。 使用以下内容:

 <xsl:variable name="year" select="substring($dateTime,1,4)" /> <xsl:variable name="month-temp" select="substring-after($dateTime,'-')" /> <xsl:variable name="month" select="substring-before($month-temp,'-')" /> <xsl:variable name="day-temp" select="substring-after($month-temp,'-')" /> <xsl:variable name="day" select="substring($day-temp,1,2)" /> <xsl:variable name="time" select="substring-after($dateTime,'T')" /> <xsl:variable name="hh" select="substring($time,1,2)" /> <xsl:variable name="mm" select="substring($time,4,2)" /> <xsl:variable name="ss" select="substring($time,7,2)" /> <xsl:value-of select="concat($month,'/',$day,'/',$year,' ',$hh,':',$mm,':',$ss)" /> 

谢谢,这个post帮了很大忙。

我正在改变使用以下date格式的RSS源: 星期一,04四月2011 23:18:00 -0700 。 这是我用来parsing它的命名模板。

 <!--Parse date format: Mon, 04 Apr 2011 23:18:00 -0700--> <xsl:template name="formatDate"> <xsl:param name="dateIn" /> <xsl:variable name="day" select="substring($dateIn, 0, 3)" /> <xsl:variable name="date" select="substring($dateIn, 6, 2)" /> <xsl:variable name="month" select="substring($dateIn, 9, 3)" /> <xsl:variable name="year" select="substring($dateIn, 13, 4)" /> <xsl:variable name="hour" select="substring($dateIn, 18, 2)" /> <xsl:variable name="min" select="substring($dateIn, 21, 2)" /> <xsl:variable name="sec" select="substring($dateIn, 24, 2)" /> <xsl:value-of select="concat($date, ' ', $month, ' ', $year, ' ', $hour, ':', $min, ':', $sec)" /> </xsl:template>