0

在这篇文章的 PS 中,您可以找到一个 C# 代码片段,它使用 XSLT 转换 + RegEx 匹配从 XML 文档中获取货币汇率值。请阅读我的问题内联引用

//+

...

//-

注释行组。

谢谢你。

PS代码:

   string currencyCode = "RUB";
var xml = new StringReader( 
    @"<gesmes:Envelope 
        xmlns:gesmes='http://www.gesmes.org/xml/2002-08-01' 
        xmlns{{EmptyNameSpace}} ='http://www.ecb.int/vocabulary/2002-08-01/eurofxref'>
        <gesmes:subject>Reference rates</gesmes:subject>
        <gesmes:Sender>
        <gesmes:name>European Central Bank</gesmes:name>
    </gesmes:Sender>
    <Cube>
        <Cube time='2012-06-27'>
        <Cube currency='USD' rate='1.2478' />
        <Cube currency='RUB' rate='41.1252' />
        <Cube currency='ZAR' rate='10.4601' />
        </Cube>
    </Cube>
    </gesmes:Envelope>"
            .Replace("{{EmptyNameSpace}}", ":ns1")
            //+
            // I'd like to get this code working the same way as it does now
            // producing
            //
            // Result: EUR/RUB => 41.1252
            //
            // output but when the above code line will be commented
            // and the below code line uncommented
            //-
            //.Replace("{{EmptyNameSpace}}", "") 
);

var xslt = new XmlTextReader(new StringReader(
@"<xsl:stylesheet version='2.0' 
    xmlns:xsl='http://www.w3.org/1999/XSL/Transform'
    xmlns:gesmes='http://www.gesmes.org/xml/2002-08-01' 
    xmlns:ns1 ='http://www.ecb.int/vocabulary/2002-08-01/eurofxref'
    >

<xsl:variable name='x1' select='gesmes:Envelope/Cube/Cube/Cube[@currency=""{0}""]' />
<xsl:template match='gesmes:Envelope'>
  [<xsl:apply-templates select='$x1' />]
</xsl:template>
<xsl:template match='Cube'>
    <xsl:value-of select='@rate' />
</xsl:template>
</xsl:stylesheet>".Replace("{0}", currencyCode)  

));

var xDoc = new XPathDocument(xml);
var xTr = new System.Xml.Xsl.XslCompiledTransform(); 
xTr.Load(xslt) ;

StringBuilder sb = new StringBuilder(); 
StringWriter writer = new StringWriter(sb); 
xTr.Transform(xDoc, null, writer);

string pattern = @"\[(?'name'[0-9]*\.[0-9]*)\]";

System.Console.WriteLine(
    "Result: EUR/{0} => {1}",
    currencyCode, 
    Regex.Match(sb.ToString(), pattern)
   .Groups["name"].Value);    

// Result: EUR/RUB => 41.1252
4

1 回答 1

0

据我查看该代码可以看出,您的问题是处理 XPath 中的默认命名空间:

  • 只要{{EmptyNameSpace}}:ns1,一切都很好 - 在您的 Xml 文档中,命名空间http://www.gesmes.org/xml/2002-08-01http://www.ecb.int/vocabulary/2002-08-01/eurofxref带有前缀gesmesns1被定义;您的 Xml 文档使用诸如<gesmes:Envelope>fromgesmes和其他标识符,即<Cube>不与任何命名空间相关联的标识符。在 XSLT 代码中,您正在使用 XPath 表达式,例如 (here shorted) gesmes:Envelope/Cube/Cube/Cube,这很好,因为该表达式引用了<Envelope>使用前缀声明的命名空间中的元素gesmes,以及<Cube>没有命名空间的元素。
  • 一旦设置{{EmptyNameSpace}}为空字符串,情况就会发生变化:现在,http://www.ecb.int/vocabulary/2002-08-01/eurofxref是文档的默认命名空间。因此,<Cube>文档中的元素被认为属于该命名空间。但是,您的 XPath 表达式仍然认为<Cube>出现在 XPath 中的元素没有命名空间——它们没有任何命名空间前缀,并且 XPath 不知道默认命名空间。

作为一种解决方案,您应该在 XPath 表达式中添加命名空间前缀 - 如果<Cube>元素属于http://www.ecb.int/vocabulary/2002-08-01/eurofxref命名空间,您的 XPath 应该如下所示:

gesmes:Envelope/ns1:Cube/ns1:Cube/ns1:Cube

(当然,您的 XSLT 中的其他 XPath 表达式必须相应地进行调整。)

如果您需要对此进行任何进一步的解释,或者我的假设是否走错了路,请告诉我。

更新:如果 Xml 文档具有默认命名空间,XSLT 应该如下所示:

<xsl:stylesheet version="2.0" xmlns:xsl='http://www.w3.org/1999/XSL/Transform' xmlns:gesmes='http://www.gesmes.org/xml/2002-08-01' xmlns:ns1='http://www.ecb.int/vocabulary/2002-08-01/eurofxref'>

    <xsl:variable name='x1' select='gesmes:Envelope/ns1:Cube/ns1:Cube/ns1:Cube[@currency="RUB"]'/>

    <xsl:template match='gesmes:Envelope'>
        [<xsl:apply-templates select='$x1'/>]
    </xsl:template>

    <xsl:template match='ns1:Cube'>
        <xsl:value-of select='@rate'/>
    </xsl:template>

</xsl:stylesheet>
于 2012-06-28T19:39:44.450 回答