C# 通过XSLT以XML格式格式化日期

C# 通过XSLT以XML格式格式化日期,c#,.net,xml,datetime,xslt,C#,.net,Xml,Datetime,Xslt,当我使用XML序列化程序对日期时间进行序列化时,它是以以下格式编写的: <Date>2007-11-14T12:01:00</Date> 2007-11-14T12:01:00 当通过XSLT样式表来输出HTML时,如何格式化它?在大多数情况下,我只需要日期,而当我需要时间时,我当然不希望“有趣的t”出现在那里。日期格式在XSLT1.0中并不容易。最优雅的方法可能是用C#编写一个简短的XSLT扩展函数,用于日期格式。下面是一个例子: <?xml version=

当我使用XML序列化程序对
日期时间
进行序列化时,它是以以下格式编写的:

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

当通过XSLT样式表来输出HTML时,如何格式化它?在大多数情况下,我只需要日期,而当我需要时间时,我当然不希望“有趣的t”出现在那里。

日期格式在XSLT1.0中并不容易。最优雅的方法可能是用C#编写一个简短的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>

使用此输入文档

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

2007-11-14T12:01:00
你会得到

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

14.11.2007 
格式化日期的函数将日期值作为字符串,格式如中所述。使用.NET的DateTime结构可以免费解析任意XSD DateTime值(包括时区说明符)、时区计算和本地化输出


但是,请注意,msxml脚本扩展有一种:每次加载XSLT时,都会动态生成包含脚本代码的程序集并将其加载到内存中。由于.NET运行时的设计,无法卸载此程序集。这就是为什么必须确保XSLT只加载一次(然后缓存以供进一步重用)。在IIS内部运行时,这一点尤其重要。

以下是几个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>



其中xpath是具有标准日期时间格式的元素或属性的路径。

John Workman在其博客[1]中详细讨论了这个问题,并给出了几种解决方案。基本上,解析各个日期组件并按照您希望的顺序重新组合。对于您的情况,纯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>

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


一月
二月
前进
四月
也许
六月
七月
八月
九月
十月
十一月
十二月
您可以选择任何方式重新组合元素


[1] @

更正roy的帖子:函数中的日期将始终获得月份值。使用以下命令:



谢谢,这篇文章帮了大忙

我正在转换一个使用以下日期格式的RSS提要:2011年4月4日星期一23:18:00-0700。这是我用来解析它的命名模板

<!--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>

很抱歉对这个旧线程进行了评论,但对于像我这样发现它的其他人,如果您使用的是MS transformer,您也可以使用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">

在msxsl:script元素中编写JavaScript:

<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)方法来包含外部js文件(例如jquery库)。这可以通过在转换之前解析xsl文件服务器端并将js文件内容作为字符串添加到CDATA部分来实现。我自己也开始走这条路,但得出的结论是,如果您需要这一级别的功能,最好将其放在管道的不同部分

来源:
参考:


这对我有用。 您可以在以下位置查看其他选项:


说明您的版本和XSLT平台很可能是XSLT 1.0和.NET,这一点非常重要,因为问题带有C标记。您是否尝试使用内置函数?是的,这和我使用的方法差不多!我只是对否决票感到好奇:有技术原因吗?或者只是个人不喜欢这种方法?我投了反对票,因为msxsl:script是不需要的(参见AnthonyW的文章,这是最优雅的解决方案),而且它有严重的缺点:。XSLT扩展对象比在.NET中创建自定义XSLT函数要好得多。请尝试:)这个问题就是我提到的,在实践中通常可以轻松解决。出于性能原因,只加载一次XSLT是一种良好的做法。XSLT扩展对象有一个很大的缺点(至少到目前为止),即它们使用后期绑定调用,因此速度非常慢。(续)AnthonyW在我看来也是一个非常优雅(纯粹)的XSLT解决方案,但是,支持不同的日期格式需要做更多的工作,因为freeXSLT没有提供所有的.NET日期时间内容。您的解决方案很优雅,但我们当然不应该手工制作日期格式化例程。@Ryan:我同意,XSLT 2对日期处理有更好的支持。不幸的是,即使是现在,在HTML浏览器的安装群中对它的支持也非常少。@AnthonyWJones:这是一种严重的轻描淡写,XSLT2.0在动态语言之外非常少。其中大部分是Java,还有一些是.NET。我们没有用于XSLT2.0的libXSLT,否则XSLT2.0会将XSLT带到少数浏览器中。一旦FOSS和高效的C/C++XSLT2.0库存在,并且跨平台依赖性相当小,我们将看到浏览器支持。你的密码让我省去了很多麻烦。
exclude-result-prefixes="msxsl js" 
<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">
<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>
<xsl:value-of select="js:javascriptFunction(string(.))"/>
<xsl:template match="date">
     <xsl:copy>
         <xsl:call-template name="formatdate">
             <xsl:with-param name="DateTimeStr" select="."/>
        </xsl:call-template>
     </xsl:copy>
  </xsl:template>

  <xsl:template name="formatdate">
     <xsl:param name="DateTimeStr" />

     <!-- input format xslt datetime string -->
     <!-- output format mm/dd/yyyy -->

     <xsl:variable name="datestr">
         <xsl:value-of select="substring-before($DateTimeStr,'T')" />
     </xsl:variable>

     <xsl:variable name="mm">
         <xsl:value-of select="substring($datestr,6,2)" />
     </xsl:variable>

     <xsl:variable name="dd">
        <xsl:value-of select="substring($datestr,9,2)" />
     </xsl:variable>

     <xsl:variable name="yyyy">
        <xsl:value-of select="substring($datestr,1,4)" />
     </xsl:variable>

     <xsl:value-of select="concat($mm,'/', $dd, '/', $yyyy)" />
  </xsl:template>