Общий XML XPath помощник с C#

Проблема- у меня есть приложение для Windows, которое загружает XML-файл, и поскольку другие входные данные используют XPath для извлечения данных, которые пользователь хочет получить из данного XML (очевидно, это не проблема). Фрагмент кода, который я использую для загрузки элемента из данного XPath, похож на

XPathDocument docNav;
XPathNavigator nav;
XPathNodeIterator NodeIter;
XmlNamespaceManager manager;

docNav = new XPathDocument(fileName);
nav = docNav.CreateNavigator();

nav.MoveToFollowing(XPathNodeType.Element);
IDictionary<string, string> names = nav.GetNamespacesInScope(XmlNamespaceScope.All);

manager = new XmlNamespaceManager(nav.NameTable);

foreach (KeyValuePair<string, string> item in names)
{
    manager.AddNamespace(item.Key, item.Value);
}

NodeIter = nav.Select(path, manager);

У меня есть один образец XML, как показано ниже:

<FpML xmlns:ns="http://www.fpml.org/2005/FpML-4-2">
  <header>
    <messageId>ts4-XYZ</messageId>
    <sentBy>XYZ</sentBy>
    <creationTimestamp>2017-07-08T08:05:53.929Z</creationTimestamp>
  </header>
  <trade>
    <tradeHeader>
      <partyTradeIdentifier>
        <partyReference href="OUR_PARTY"/>
        <tradeId tradeIdScheme="uniqueId">AAAAAA</tradeId>
      </partyTradeIdentifier>
      <partyTradeInformation>
        <partyReference href="OUR_PARTY"/>
        <trader>dummy trader name</trader>
      </partyTradeInformation>
      <tradeDate>2017-07-08</tradeDate>
      <tsfpml:completedDateTime>2017-07-08T08:05:53.656Z</tsfpml:completedDateTime>
    </tradeHeader>
  </trade>  
</FpML>

Теперь я хочу получить значение элемента 'trader'. Это работает, если я использую запрос XPath как

//*[local-name()="trader" and namespace-uri()='http://www.fpml.org/2005/FpML-4-2']

Но он не может загрузить любой элемент, если я дам /FpML/trade/tradeHeader/partyTradeInformation/trader

Что нужно изменить в моем приложении-помощнике XPath, чтобы выбрать узлы с пространством имен по умолчанию в целом.

Заранее спасибо!

2 ответа

Вы можете удалить пространства имен, используя:

        public static string RemoveAllNamespaces(string xmlDocument)
        {
            XElement xmlDocumentWithoutNs = RemoveAllNamespaces(XElement.Parse(xmlDocument));

            return xmlDocumentWithoutNs.ToString();
        }
        private static XElement RemoveAllNamespaces(XElement xmlDocument)
        {
            if (!xmlDocument.HasElements)
            {
                XElement xElement = new XElement(xmlDocument.Name.LocalName);
                xElement.Value = xmlDocument.Value;

                foreach (XAttribute attribute in xmlDocument.Attributes())
                    xElement.Add(attribute);

                return xElement;
            }
            return new XElement(xmlDocument.Name.LocalName, xmlDocument.Elements().Select(el => RemoveAllNamespaces(el)));
        }

И называется:

var doc = new XmlDocument();
doc.Load(fileName);
doc.InnerText = RemoveAllNamespaces(doc.InnerText);
var nav = doc.CreateNavigator();

Тогда у вас не так много выбора, используйте XmlNamespaceManager:

var docNav = new XPathDocument("");
var nav = docNav.CreateNavigator();
XmlNamespaceManager nsmanager = new XmlNamespaceManager(docNav.NameTable);
nsmanager.AddNamespace("x", "ns:http://www.fpml.org/2005/FpML-4-2");
var nodes = nav.Select(@"/x:FpML/trade/tradeHeader/partyTradeInformation/trader", nsmanager);

Он должен работать. Я не могу проверить себя, но если это не сработает, протестируйте изменение URI пространства имен без "ns:", или добавьте другие "x:" в свой XPath и отредактируйте мой ответ с правильным ответом;)

Другие вопросы по тегам