当命名空间不再可用时解析xml文档

时间:2009-07-10 09:50:47

标签: c# xml xpath

我需要循环使用一些相当大的复杂xml文档。 xmlns定义在文档的顶部,但是指向的url不再可用。

解析文件以使用C#从中获取重要数据的最佳方法是什么?

我尝试将其加载到数据集中但偶尔会收到错误: 表(端点)不能是嵌套关系中的子表。 要么 无法将SimpleContent列添加到包含元素列或嵌套关系的表中。

XPath是我的下一个调用端口,但由于缺少命名空间,我遇到了问题。

我怀疑这严重限制了我的选择,但有人有任何建议吗?

XML文档的片段:

<?xml version="1.0" encoding="UTF-8"?>
<cdr:cdr_set xmlns:cdr="http://www.naturalconvergence.com/schema/cdr/v3/cdr">

<!--  Copyright (c) 2001-2009, all rights reserved  -->

<cdr:cdr xmlns:cdr="http://www.naturalconvergence.com/schema/cdr/v3/cdr">
  <cdr:call_id>2040-1247062136726-5485131</cdr:call_id>
  <cdr:cdr_id>1</cdr:cdr_id>
  <cdr:status>Normal</cdr:status>
  <cdr:responsibility>
    <cdr:tenant id="17">
      <cdr:name>SpiriTel plc</cdr:name>
    </cdr:tenant>
    <cdr:site id="45">
      <cdr:name>KWS</cdr:name>
      <cdr:time_zone>GB</cdr:time_zone>
    </cdr:site>
  </cdr:responsibility>
  <cdr:originator type="sipGateway">
    <cdr:sipGateway id="3">
      <cdr:name>Audiocodes-91</cdr:name>
    </cdr:sipGateway>
  </cdr:originator>
  <cdr:terminator type="group">
    <cdr:group>
      <cdr:tenant id="17">
        <cdr:name>SpiriTel plc</cdr:name>
      </cdr:tenant>
      <cdr:type>Broadcast</cdr:type>
      <cdr:extension>6024</cdr:extension>
      <cdr:name>OLD PMS DDIS DO NOT USE</cdr:name>
    </cdr:group>
  </cdr:terminator>
  <cdr:initiation>Dialed</cdr:initiation>
  <cdr:calling_number>02087893850</cdr:calling_number>
  <cdr:dialed_number>01942760142</cdr:dialed_number>
  <cdr:target>6024</cdr:target>
  <cdr:direction>Inbound</cdr:direction>
  <cdr:disposition>No Answer</cdr:disposition>
  <cdr:timezone>GB</cdr:timezone>
  <cdr:origination_timestamp>2009-07-08T15:08:56.727+01:00</cdr:origination_timestamp>
  <cdr:release_timestamp>2009-07-08T15:09:26.493+01:00</cdr:release_timestamp>
  <cdr:release_cause>Normal Clearing</cdr:release_cause>
  <cdr:call_duration>PT29S</cdr:call_duration>
  <cdr:redirected>false</cdr:redirected>
  <cdr:conference>false</cdr:conference>
  <cdr:transferred>false</cdr:transferred>
  <cdr:estimated>false</cdr:estimated>
  <cdr:interim>false</cdr:interim>
  <cdr:segments>
    <cdr:segment>
      <cdr:originationTimestamp>2009-07-08T15:08:56.727+01:00</cdr:originationTimestamp>
      <cdr:initiation>Dialed</cdr:initiation>
      <cdr:call_id>2040-1247062136726-5485131</cdr:call_id>
      <cdr:originator type="sipGateway">
        <cdr:sipGateway id="3">
          <cdr:name>Audiocodes-91</cdr:name>
        </cdr:sipGateway>
      </cdr:originator>
      <cdr:termination_attempt>
        <cdr:termination_timestamp>2009-07-08T15:08:56.728+01:00</cdr:termination_timestamp>
        <cdr:terminator type="group">
          <cdr:group>
            <cdr:tenant id="17">
              <cdr:name>SpiriTel plc</cdr:name>
            </cdr:tenant>
            <cdr:type>Broadcast</cdr:type>
            <cdr:extension>6024</cdr:extension>
            <cdr:name>OLD PMS DDIS DO NOT USE</cdr:name>
          </cdr:group>
        </cdr:terminator>
        <cdr:provided_address>01942760142</cdr:provided_address>
        <cdr:direction>Inbound</cdr:direction>
        <cdr:disposition>No Answer</cdr:disposition>
      </cdr:termination_attempt>
    </cdr:segment>
  </cdr:segments>
</cdr:cdr>

...

</cdr:cdr_set>

每个条目基本相同,但有时会出现差异,例如某些字段可能会丢失,如果不是必需的话。

2 个答案:

答案 0 :(得分:5)

xml文件中的这些值是标识符,而不是定位器。除非您希望下载架构,否则根本不需要它,并且如果需要可以“轻松”。我希望最好将其加载到XmlDocument / XDocument并尝试访问数据。

例如:

XmlDocument doc = new XmlDocument();
doc.Load("cdr.xml");
XmlNamespaceManager ns = new XmlNamespaceManager(doc.NameTable);
ns.AddNamespace("cdr", "http://www.naturalconvergence.com/schema/cdr/v3/cdr");
XmlElement el = (XmlElement)doc.SelectSingleNode(
    "cdr:cdr_set/cdr:cdr/cdr:originator", ns);
Console.WriteLine(el.GetAttribute("type"));

或循环遍历cdr元素:

    foreach (XmlElement cdr in doc.SelectNodes("/cdr:cdr_set/cdr:cdr", ns))
    {
        Console.WriteLine(cdr.SelectSingleNode("cdr:call_id", ns).InnerText);
    }

请注意,文档中使用的别名与XmlNamespaceManager中使用的别名基本无关,因此您需要重新声明它。我可以很容易地使用x作为我在C#中的别名。


当然,如果您更喜欢使用对象模型;通过xsd运行它(其中cdr.xml是您的示例文件):

xsd cdr.xml
xsd cdr.xsd /classes

现在您可以使用XmlSerializer加载它。

答案 1 :(得分:1)

alternativley将它加载到Xdocument并使用linq2XML? ......虽然你可能会得到同样的错误。

我不知道你想要什么数据,所以很难建议查询。 在大多数情况下,我个人更喜欢将XDocument用于xmlDocument。

自动生成XSD的唯一问题是,如果您没有使用大小合适的样本数据,它可能会使您的数据类型出现严重错误。