Общий 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 и отредактируйте мой ответ с правильным ответом;)